Skip to content

Reverse engineering the bluetooth protocol used by the QiYi Smart Cube

Notifications You must be signed in to change notification settings

Flying-Toast/qiyi_smartcube_protocol

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

17 Commits
 
 
 
 

Repository files navigation

Video.mov

The QiYi Smart Cube is the cheapest of the newly invented genre of bluetooth-enabled "smart" Rubik's cubes. Unfortunately QiYi has refused to publish the protocol used by the cube, and until now there hasn't been much progress in reverse engineering it. This document provides a best effort to reverse engineer and document the protocol, but it is not a complete specification. If you discover anything new please send a pull request!

This document assumes you are somewhat familiar with Bluetooth Low Energy/GATT. If you are new to it, I recommend reading this introductory article.

This repo also contains a Wireshark plugin that is helpful when doing any work with the cube protocol.

GATT Profile

The protocol is on top of Bluetooth Low Energy. The cube has the following GATT profile:

  • Service 1801
    • Characteristic 2a05
  • Service fff0
    • Characteristic fff4
    • Characteristic fff5
    • Characteristic fff6
    • Characteristic fff7
  • Service 5833ff01-9b8b-5191-6142-22a4536ef123
    • Characteristic 5833ff02-9b8b-5191-6142-22a4536ef123
    • Characteristic 5833ff03-9b8b-5191-6142-22a4536ef123

A lot of these use the standard base uuid, but as far as I can tell they don't follow the standards and might as well be custom uuids. The majority of the protocol just uses WRITEs and NOTIFYs on the fff6 characteristic.

Protocol

app->cube messages are sent by performing a WRITE of the data to the fff6 characteristic.

cube->app messages are received via NOTIFY events on the fff6 characteristic.

Encryption

All messages sent to/received from the cube are encrypted using AES128 in ECB mode with the fixed key 57b1f9abcd5ae8a79cb98ce7578c5108 ([87, 177, 249, 171, 205, 90, 232, 167, 156, 185, 140, 231, 87, 140, 81, 8])

Before being encrypted, messages are padded with trailing zeros until the total message length is a multiple of 16.

For the rest of this document, all values are given in their decrypted form and it is implied that messages being sent will be encrypted and messages received have been decrypted. It is also implied that the checksum is included in each message.

Messages

All messages (both app->cube and cube->app) start with the byte 0xfe. The next byte is the length of the message (excluding padding).

Additionally, for all cube->app messages:

  • The byte after the length is an "opcode" that specifies type type of message.
  • After the opcode is a 4 byte big-endian timestamp. The timestamp is in units of 1.6 milliseconds! Divide it by 1.6 to convert to milliseconds!

These are the kinds of messages:

Checksum

The last 2 bytes of each message (before the zero padding) are a checksum of the message (minus any zero padding) using the CRC-16-MODBUS algorithm. The checksum is in little-endian. Example: fe 09 02 00 02 45 2c ef 1b Here, the bolded part (ef 1b) is the little-endian checksum of fe 09 02 00 02 45 2c. So for this example the checksum is 0x1bef.

App Hello

Command Direction
App Hello app->cube

Immediately after connecting to the cube, you need to write an "App Hello" message to the fff6 characteristic. The App Hello must be the first thing you send to the cube. The cube won't reply to anything you send unless you've already performed the App Hello.

The MAC address field needs to be reversed; the following example is for a cube with the address cc a3 00 00 25 13:

L = length
A = cube MAC address (but the bytes are backwards!)
C = checksum

   L                ??                         A           C
   /\ /------------------------------\ /---------------\ /---\
fe 15 00 6b 01 00 00 22 06 00 02 08 00 13 25 00 00 a3 cc XX XX
Bytes (start index, length) Type Description
1, 1 u8 Length (always 21 for App Hello)
2, 11 ? Unknown, but doesn't seem to matter what data is in here; you can just fill it with zeros.
13, 6 - The MAC address of the cube, backwards
19, 2 u16_le Checksum

Message Acknowledgement

Command Direction
ACK app->cube

Upon receiving most cube->app messages, you have to send an ACK message back to the cube. This is the ACK format:

L = length (always 9 for ACKs)
H = bytes 3-7 of the message being ACKed
C = checksum

   L       H           C
   /\ /------------\ /---\
fe 09 XX XX XX XX XX CC CC

That would be an ACK for a message that looks like this:

            H
      /------------\
fe zz XX XX XX XX XX zz zz zz zz zz zz ...

Not all types of cube->app messages need to be ACKed all the time - see the "Needs ACK?" section in the respective command's descriptions.

Cube Hello

Command Direction Needs ACK?
Cube Hello cube->app yes

The "Cube Hello" message is sent by the cube immediately after it receives the App Hello.

L = length (38)
O = opcode (0x2)
TS = timestamp (units of 1.6ms)
S = initial cube state
B = battery level
C = checksum

   L  O       TS                                         S                                            ?  B    C
   /\ /\ /---------\ /------------------------------------------------------------------------------\ /\ /\ /---\
fe 26 02 00 0e 2d aa 33 33 33 33 13 11 11 11 11 44 44 44 44 24 22 22 22 22 00 00 00 00 50 55 55 55 55 00 64 XX XX
Bytes (start index, length) Type Description
1, 1 u8 Length (always 38 for Cube Hello)
2, 1 u8 Opcode (0x2 for Cube Hello)
3, 4 u32_be Timestamp (units of 1.6ms)
7, 27 CubeState Initial cube state
34, 1 ? Unknown
35, 1 u8 Battery level (between 0 and 100)
36, 2 u16_le Checksum

State Change

Command Direction Needs ACK?
State Change cube->app Only if "needs ACK" byte is set to 1

The only time you need to send an ACK for a State Change is when the "needs ACK" byte (the byte at index 91) is set to 1. For all other State Change messages, no ACK is needed.

Most State Change messages do not need to be ACKed (and thus will have their "needs ACK" byte set to 0). The "needs ACK" field only becomes 1 when the CubeState field for this message is a solved state. However: there is a glitch that can occur if the cube is solved while doing fast slice moves (e.g. an H-perm), in which case the cube is physically in a solved state but it never sends a State Change for that solved state. In this case, the cube will send a State Change message that has the "needs ACK" byte set to 1, but where its CubeState field is not solved. To avoid this issue, treat any State Change messages where "needs ACK" is 1 as if the CubeState is the solved state, even if it isn't.

L = length (94)
O = opcode (0x3)
TS = timestamp (units of 1.6ms)
S = cube state
T = what turn was done to the cube
B = battery level out of 100
P = previous timestamps + turns
A = needs ACK?
C = checksum

   L  O      TS                                          S                                            T  B                                                                             P                                                                                         A    C
   /\ /\ /---------\ /------------------------------------------------------------------------------\ /\ /\ /------------------------------------------------------------------------------------------------------------------------------------------------------------------\ /\ /---\
fe 5e 03 00 06 98 e5 33 33 33 33 13 11 11 11 11 44 44 44 44 24 22 22 22 22 00 00 00 00 50 55 55 55 55 08 64 ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff ff 00 03 00 c6 01 00 03 04 ee 02 00 06 94 ab 07 01 XX XX
Bytes (start index, length) Type Description
1, 1 u8 Length (always 94 for State Change)
2, 1 u8 Opcode (0x3 for State Change)
3, 4 u32_be Timestamp (units of 1.6ms)
7, 27 CubeState Cube state
34, 1 u8 The move that was applied to the cube to bring it into this state. See the table below.
35, 1 u8 Battery level, 0-100
36, 55 - List of previous moves+their timestamps
91, 1 u8 needs ACK
92, 2 u16_le Checksum

Looking at the cube with white on top and green in front, this is how numbers correspond to turns:

Byte Move
0x1 L'
0x2 L
0x3 R'
0x4 R
0x5 D'
0x6 D
0x7 U'
0x8 U
0x9 F'
0xa F
0xb B'
0xc B

Sync State

Command Direction
Sync State app->cube

If the physical state of the cube becomes out of sync with what the cube thinks it is, you can send the Sync State command to tell the cube to reset its remembered state to the one you provide in the command. When the cube recieves a Sync State command it will reply with a Sync Confirmation command.

L = length (38)
S = new state to set the cube into
C = checksum

   L        ?                                                S                                          ?     C
   /\ /------------\ /------------------------------------------------------------------------------\ /---\ /---\
fe 26 04 17 88 8b 31 33 33 33 33 13 11 11 11 11 44 44 44 44 24 22 22 22 22 00 00 00 00 50 55 55 55 55 00 00 XX XX
Bytes (start index, length) Type Description
1, 1 u8 Length (always 38 for Sync State)
2, 5 ? Unknown
7, 27 CubeState The state to set the cube to
34, 2 ? Unknown
36, 2 u16_le Checksum

Sync Confirmation

Command Direction Needs ACK?
Sync Confirmation cube->app no

Sent in response to a Sync State command.

L = length (38)
O = opcode (0x4)
TS = timestamp (units of 1.6ms)
S = cube's current state
B = battery level
C = checksum

   L  O      TS                                               S                                       ?  B    C
   /\ /\ /---------\ /------------------------------------------------------------------------------\ /\ /\ /---\
fe 26 04 00 00 df cc 33 33 33 33 13 11 11 11 11 44 44 44 44 24 22 22 22 22 00 00 00 00 50 55 55 55 55 00 64 XX XX
Bytes (start index, length) Type Description
1, 1 u8 Length (always 38 for Sync Confirmation)
2, 1 u8 Opcode (0x4 for Sync Confirmation)
3, 4 u32_be Timestamp (units of 1.6ms)
7, 27 CubeState State the cube now thinks it's in
34, 1 ? Unknown
35, 1 u8 Battery level, 0-100
36, 2 u16_le Checksum

Cube State Format

Cube states are stored as a 54-item-long array of 4-bit numbers, where each 4-bit number represents the color of a facelet (see table below). The index of the item in the array tells you where on the cube the facelet is.

Number Color
0 orange
1 red
2 yellow
3 white
4 green
5 blue

Within each byte, the lower 4 bits are ordered before the upper 4 bits. This is true for all the bytes but is especially noticeable with the byte on the boundry between two faces (see the "!!" in diagram below).

A solved cube looks like this:

    WHITE   !!     RED        GREEN    !!   YELLOW       ORANGE   !!    BLUE
/----------\!!/----------\ /----------\!!/----------\ /----------\!!/----------\
33 33 33 33 13 11 11 11 11 44 44 44 44 24 22 22 22 22 00 00 00 00 50 55 55 55 55

Indices into each color's array map to faclets like this (W=white, O=orange, etc):

          ┌──┬──┬──┐
          │W0│W1│W2│
          ├──┼──┼──┤
          │W3│W4│W5│
          ├──┼──┼──┤
          │W6│W7│W8│
          └──┴──┴──┘
┌──┬──┬──┐┌──┬──┬──┐┌──┬──┬──┐┌──┬──┬──┐
│O0│O1│O2││G0│G1│G2││R0│R1│R2││B0│B1│B2│
├──┼──┼──┤├──┼──┼──┤├──┼──┼──┤├──┼──┼──┤
│O3│O4│O5││G3│G4│G5││R3│R4│R5││B3│B4│B5│
├──┼──┼──┤├──┼──┼──┤├──┼──┼──┤├──┼──┼──┤
│O6│O7│O8││G6│G7│G8││R6│R7│R8││B6│B7│B8│
└──┴──┴──┘└──┴──┴──┘└──┴──┴──┘└──┴──┴──┘
          ┌──┬──┬──┐
          │Y0│Y1│Y2│
          ├──┼──┼──┤
          │Y3│Y4│Y5│
          ├──┼──┼──┤
          │Y6│Y7│Y8│
          └──┴──┴──┘

About

Reverse engineering the bluetooth protocol used by the QiYi Smart Cube

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages