Me: Ok, Google. Stand up.
Google Assistant: Ok. Raising your Jarvis desk.
Desk: begins moving obediently
Notice: This project probably will void your warranty. Proceed at your own risk.
This is a project to add a computer interface to my Jarvis stand-up desk from fully.com. I reverse-engineered the interface between the desk handset and the desk controller box to find out how they communicate. Then I wired an ESP8266 wifi module (like the Huzzah from Adafruit) to a couple of RJ-45 jacks and plugged it all together. Now I can snoop on the messages being sent between the desk and the handset, inject my own messages to either one, and monitor the desk position from the internet using the AdafruitIO MQTT dashboard interface.
It's pretty cool, but honestly, my motivation was to be able to lower the desk without getting up to see the handset controller. The desk sits next to my couch at home. When I want to set a drink on it while watching TV, it's inconvenient if I left it standing up. "Ok, Google. Sit down" The desk lowers, and I can put my drink down and continue watching Netflix. To be fair, I didn't need to decode the serial interface for this because the buttons used for memory presets are sent on separate signal lines. The serial protocol decoding was just for fun.
As a useless side-benefit, you can look back at my desk's height history and see every time it moved up or down and what preset was used in the last 30 days. And I can control my desk from anywhere on the internet, so I can lower it even if I'm not home, though that seems like a bad idea.
Features so far:
- Reverse engineered the UART interface and most control messages
- Snoop on the messages being sent between the desk and the handset
- Read the handset buttons
- Telnet wifi interface for debug logging
- Simulate handset buttons (e.g. press 3)
- Monitor desk configuration
- Connected the desk to AdafruitIO's MQTT dashboard for monitoring
- Can control the desk position via the AdafruitIO dashboard
- Interfaced IFTTT to the AdafruitIO feed controlling my desk
- Added a Google Assistant command phrase to raise and lower my desk
Future things:
- Inject control messages to the UART to control and configure the desk
- Inject control messages to the UART to control the handset display
- Explore the handset to see what else it might be hiding
- Keep the handset display awake so buttons are always visible[1]
- Explore the other UART control protocol the desk sometimes falls into
- Expose all handset controls so people can add memory presets to their cheap-model handsets.
- Expand the telnet interface to permit different log levels, message injection, etc.
[1] Sadly, you can't display random text on your display module, as far as I know. But you could make a nice little 3-digit clock or something there, if you wanted.
You will need to edit the local-config.h
file to include your own WiFi access credentials and your Adafruit IO
username and API key.
// AdafruitIO credentials
#define IO_USERNAME "your-adafruitio-id"
#define IO_KEY "your-adafruitio-api-key-hex"
// Wifi AP credentials
#define WIFI_SSID "wifi-ssid"
#define WIFI_PASS "wifi-password"
For your safety (and mine) this file is in the .gitignore
file for this project to help prevent us from accidentally
committing password information to the project and sharing it with the world. If you want to keep this config file
history in your own local git repository, please be aware of this limitation.
You will need to edit the pin definitions in jarvis_pinouts.h
to match the GPIOs you used on your board. You will need
6 pins defined. You can use one of the existing layouts defined there by uncommenting the correct #define directive, or
you may choose your own pin layout by adding your own customized version with the DEFINE_PINS
macro.
This code should be able to build with the Arduino IDE or the arduino-cli tool. You will need extra libraries installed to support your target board. This code is written for the ESP8266, so to build it as-is, you would need the ESP8266 toolkit, for example. If you use the Adafruit.IO feed you will also need to install the Adafruit IO library and its dependencies.
To build with arduino-cli, do this:
arduino-cli compile -b esp8266:esp8266:oak .
The code here supports Over-the-Air (OTA) program upload using the ESP8266 "espota.py" tool or the ArduinoIDE. Once a version of this code is booted and running on the board, future updates can be uploaded with the OTA update.
In the Arduino IDE, OTA upload can be done by changing the port from the serial port to the Jarvis (network) port here: ArduinoIDE > Tools > Port : Jarvis If the device is online and connected to your network, it should show up in the Port menu under "Network ports".
To upload using the espota utility:
$HOME/.arduino15/packages/esp8266/hardware/esp8266/3.0.2/tools/espota.py -i jarvis.local -p 8266 -f Jarvis.ino.bin
The code runs a telnet interface for debug output. This is mostly for convenience, but it also frees up the serial port pins for use as GPIO pins to connect to the desk.
Telnet to jarvis.local
on port 23 to see the debug logs.
telnet jarvis.local
WARNING: This is the part that probably voids your warranty.
The 6 GPIO pins you chose on the ESP8266 board should be wired to the RJ-45 signal pins on your desk. In addition, you will need to wire the RJ-45 GND pin (pin 3) to your controller ground, and for convenience you can wire the RJ-45 VCC (pin 5) to your controller's VCC to power it when it's not connected to external power.
See this discussion for notes on how I wired up my board.
The Jarvis desk handset style I have is the touchpanel interface with memories. You can surely use the desk without this interface if you design your own, but you need to know the protocols. I will try to publish what I discovered through reverse-engineering here.
- Handset model: JCHT35M16-1
- Desk controller model: FullyCB2C-A
Note: Other desk owners have mentioned that the simple 2-button handset interface uses a different desk controller (model: JCB35N2) but has the same electrical interface. However the serial interface uses a different protocol. You can find some details in the JCB35N2 controller protocol discussion. It's a bit exciting that the basic and advanced models both seem to support both protocols in some form, but so far I don't know how to switch between them on either controller.
The interface from the handset to the controller is via an RJ-45 8-pin connector (like an ethernet connector). The pins in the RJ-45 jack are numbered 1 to 8.
Note: I don't know any official names for these signal lines, so I made these up myself. Any mistakes here about the meaning or use of these lines are mine.
Following are their usage on the desk:
Pin | Label | Description |
---|---|---|
1 | HS3 | Handset control line 3 [1] |
2 | DTX | Serial control messages from controller to handset [2] |
3 | GND | Ground |
4 | HTX | Serial control messages from handset to controller [2] |
5 | VCC | Vcc (5vdc) supply from desk controller [3] |
6 | HS2 | Handset control line 2 [1] |
7 | HS1 | Handset control line 1 [1] |
8 | HS0 | Handset control line 0 [1] |
[1] The pins (HS0, HS1, HS2, HS3) make up the "button-press" mux lines. See HSx control lines below
[2] Serial port UART is 9600bps, 8 data bits, No parity, 1 stop bit
[3] 5VDC supply appears capable of driving at least 300ma
[4] All the signal lines have 5v pullups provided by the desk controller
In this capture, I pressed "Memory" to wake the screen up, and then 3 to move the desk to position 3.
The desk provides 5v pullups on all six signal lines. A "mark" is signaled by pulling the line low.
The ESP8266 microcontroller is only capable of outputting 3.3v on its GPIOs. But I don't need to pull any of these pins high to 5v because the desk does that for me. I only need to pull these signals low to 0.0v.
It is safe to sink 5v on these GPIO pins on the ESP8266, so long as the carrier board doesn't mind the extra voltage. Opinions vary on this topic, though. You may feel safer using a level shifter or some transistor or diode assistance to pull these 5v signals low on your board. Doing this is beyond my expertise, though, so I'm afraid I can't help you with this part if you choose to go this route.
On the mechanical desk control buttons, when you press the up button, one of the HS[0-2] lines is pulled low. Similarly when you press the down button, another line is pulled low. When you use a smart controller interface like the touch-screen model, multiple lines may be pulled low simultaneously to indicate up to 7 different buttons. If you combine the first three lines' inverted logical signals, you will find a "button number" between 0 and 7 is possible.
Zero means no button is being pressed. This is "open".
Note: HS3 seems to be a special line used only for Memory Button signaling.
Down Up 1 2 3 4 M
HS3 X
HS2 X X X
HS1 X X X
HS0 X X X X
Translated into binary, these buttons send these codes:
Button | Value | Notes |
---|---|---|
"Open" | 0 | |
Down | 1 | |
Up | 2 | |
1 | 3 | |
2 | 4 | |
3 | 5 | |
4 | 6 | |
M | 9 | (always sent as two 30ms pulses) |
The Jarvis UART signaling protocol sends checksummed control packets consisting of between 6 and 9 bytes. All data is sent using 9600 bps, 8 data bits, no parity (9600/8N1). The desk controller sends data on DTX; the handset sends data on HTX. Command packets are the same format on both.
[address] [command] [length] [params] [checksum] [eom]
Field | Octets | Description |
---|---|---|
ADDRESS | 2 | Same value in both bytes; 0xF1 for Handset, 0xF2 for controller |
COMMAND | 1 | See KNOWN COMMANDS |
LENGTH | 1 | Length of PARAMS bytes that follow (0 to 3 (??)) |
PARAMS | LENGTH | Omitted if LENGTH=0 |
CHECKSUM | 1 | 8-bit sum of {COMMAND, LENGTH, PARAMS} |
EOM 0x7E | 1 | End of message (0x7E) |
The address byte is repeated two times. The desk controller always sends 0xF2 0xF2. The handset always sends 0xF1 0xF1.
The PARAMS field consists of 0 to 3 bytes. I haven't decoded all the meaning here. WIP
The checksum is the low-byte sum of the all the payload bytes. In pseudocode,
CHECKSUM = sum(COMMAND, LENGTH, PARAMS) % 0xFF
The EOM byte is always 0x7E.
I will use a shorthand to refer to these packets later. The desk controller packets will be
given as CONTROLLER(COMMAND, PARAMS, ...)
and the handset packets will be given as
HANDSET(COMMAND, PARAMS...)
. PARAMS
are given as P0
for the first byte, P1
for the second,
etc. 16-bit params are indicated like {P0,P1}
. Optional params are shown in brackets.
For example, the HANDSET(0x29)
message is sent from the handset to the desk. It represents
this sequence of 6 bytes:
F1 F1 29 00 29 7E
ADDRESS COMMAND LENGTH CHECKSUM EOM
Smart handset sends a NULL (0x00)
byte on first startup. If no response is received from the
desk controller, it sends a 230ms BREAK signal followed by another NULL
byte. If still no
response is received, it begins sending HANDSET(0x29)
repeatedly.
Note: I don't know any official names for these commands, so I made these up myself. Any mistakes here about the command usage or meaning are mine. All transmitted byte values are given in hex.
Name | CMD | Desk/HS | Params | Description |
---|---|---|---|---|
HEIGHT | 01 | D | 3 | Height report; P0=4 (mm?) |
{P0,P1} = height in mm or tenths of inches | ||||
P2 = ??? (0x7 or 0xF seen so far) Not units | ||||
Height from 240..530 is in inches | ||||
Height from 650..1290 is in mm | ||||
PROGMEM_1 | 03 | H | 0 | Set memory position 1 to current height |
PROGMEM_2 | 04 | H | 0 | Set memory position 2 to current height |
PROGMEM_3 | 25 | H | 0 | Set memory position 3 to current height |
PROGMEM_4 | 26 | H | 0 | Set memory position 4 to current height |
UNITS | 0E | H | 1 | Set units to cm/inches |
P0=0x00 Set units to CM | ||||
P0=0x01 Set units to IN | ||||
MEM_MODE | 19 | H | 1 | Set memory mode |
P0=0x00 One-touch mode | ||||
P0=0x01 Constant touch mode | ||||
COLL_SENS | 1D | H | 1 | Set anti-collision sensitivity (Sent 1x; no repeats) |
P0=0x01 High | ||||
P0=0x02 Medium | ||||
P0=0x03 Low | ||||
LIMIT_RESP | 20 | D | 1 | Max-height set/cleared; response to [21]; |
P0=0x00 "Min/max cleared" | ||||
P0=0x01 "Max-height set" | ||||
P0=0x10 "Min-height set" | ||||
SET_MAX | 21 | H | 0 | Set max height; Sets max-height to current height |
21 | D | 2 | Report max-height; [P0,P1] = max-height | |
Q: Can we set max-height by giving params? | ||||
SET_MIN | 22 | H | 0 | Set min height; Sets min-height to current height |
22 | D | 2 | Report min-height; [P0,P1] = min-height | |
LIMIT_CLR | 23 | H | 1 | Clear min/max height |
P0=0x01 Max-height cleared | ||||
P0=0x02 Min-height cleared | ||||
LIMIT_STOP | 23 | D | 1 | Min/Max reached |
P0=0x01 "Max-height reached" | ||||
P0=0x02 "Min-height reached" | ||||
WAKE | 29 | H | 0 | Poll message sent when desk doesn't respond to BREAK messages |
RESET | 40 | D | 0 | Indicates desk in RESET mode; Displays "RESET" |
CALIBRATE | 91 | H | 0 | Height calibration (Repeats 2x) |
Desk must be at lowest position | ||||
Desk enters RESET mode after this | ||||
REP_PRESET | 92 | D | 1 | Moving to Preset location |
P0=[4,8,10,20] mapping to presets [1,2,3,4] |
The desk controller responds to the handset by sending a display message containing the current desk height. It sends this several dozen times while the display is on. The height is encoded as a two-byte word (HI, LOW) which make up the desk's height. The units are in millimeters or tenths of inches depending on the configuration UNITS setting. The packet looks like this:
CONTROLLER(0x01, HI, LOW, 0x07)
For example, to indicate 128.6cm (1286mm), the desk sends CONTROLLER(0x01, 0x05, 0x06, 0x07)
,
because 0x0506 = 1286 in decimal. I don't know what the third parameter is here (0x07). Sometimes
it's 0x0F. More experimenting is needed.
Memories are programmed by sending one of the PROGMEM_*
commands to the controller. Each
command takes no parameters but sets the memory to the current height of the desk.
The desk does not acknowledge the setting.
When a Preset button is pressed (not in a programming mode) the lines HS0
..HS3
are signaled as
shown above in HSx control lines. Notice that the binary pattern flagged in those pins maps
preset button 1 to 0x03, 2 to 0x04, 3 to 0x05 and 4 to 0x06.
The command codes to set these presets are mapped similarly, but not identically:
PROGMEM_1 = 0x03, PROGMEM_2 = 0x04, PROGMEM_3 = 0x25 and PROGMEM_4 = 0x26
Why not the same? Is it because buttons 3 and 4 were added on later, but commands 0x05/0x06 were already used? Is bit 5 (0x20) used to signal something special?
See the table above for the short version. Each command may or may not be sent multiple times and have a different stream of responses from the other side. I'll try to document my findings about these anomalies here later.
Robodesk Jarvis v1 was built with a DigiStump Oak. I don't recommend these boards since they seem to have stopped making them a few years ago. But I leave these notes here for myself:
Install using the ESP8266 Community package here: https://arduino.esp8266.com/stable/package_esp8266com_index.json
First time programming: Connect P2 to GND to start Oak in serial program mode
Thereafter, use the OTA Software Upload described above.