The driver station is a program created by FIRST that allows teams to control their robot wirelessly through a 2.4ghz Wifi radio. Normally, the driver station is allowed only to work with the RoboRIO and its own Driver station class, which handles the UDP frames by itself. The software in this repo is what is needed to make an ESP12F module along with a HERO development board to talk with the Driver station and provide enough functionality to get a robot to drive using the driver station.
- Ensure HERO is imaged with latest firmware, see below for instructions on flashing the latest firmware
- Connect ESP Module to port 1 of HERO
- Deploy HERO_ESP_Writer project to HERO
- Keep an eye on the logs to ensure the file is written
- Once the Module is flashed, connect to its WiFi. The password is "password1"
- Set computer to static IP (Process below)
- Deploy HERO_DriverStationExample to HERO
- Open FRC Driver station
- Put in team 33
- You're good to go
This is what a successful module flash looks like
- Right click on wireless connections, the button you pressed to connect to the module
- Click on Network and Sharing Center
- Click on the network in the middle right next to Connections
- Click on Properties
- Click on Internet Protocol Version 4 (TCP/IPv4) and click Properties
- Check Use the following IP address
- IP address: 10.0.33.5
- Subnet mask: 255.0.0.0
- Default gateway: Blank
- Click OK and exit out of everything
- Connect HERO to computer using A to A USB cable
- While connecting HERO to computer, hold down the button on the HERO putting it into bootloader mode
- Open Lifeboat imager
- Select correct firmware, at the time of this writing it was 0.11.2.0.crf
- Press Image HERO
For more information on flashing the HERO, go here to the user guide
- Selecting mode robot is in, ie Teleop Enabled, Auton disabled, Test disabled
- Enable/Disable safety, Watchdog is automatically fed if enabled, otherwise it isn't
- Communications and Robot code indicators work as expected
- Multiple joysticks
- Round trip time graph on Driver Station
- Battery voltage can be sent to driver station
- UDP use, set a port and send a byte array of data to be used for a custom dashboard
- Bin File - The .bin file for driver station functionality
- BinSplitter - The project that takes a bin file and splits it into 11 25kb files that the HERO can use
- Documentation - Folder for pictures and files that show how this was created
- ESP_DriverStation_Source - Arduino source code for ESP module that enables driver station
- HERO_DriverStationExample - Example project for enabling the driver station and controlling a robot using Arcade drive
- HERO_ESP_Writer - Project that flashes the bin files to the ESP module
All the hardware needed for this functionality is the HERO board and an ESP12F module along with a ribbon cable connecting the two. Once you flash the module once it is possible to perform an Over the Air (OTA) update. Software wise, a new firmware image needs to be flashed onto the ESP module, but the ability to do so is done through the HERO itself, and an example project is set up to work with the new driverstation class.
Look at Quick Start Guide above
Connect to the module over wifi and find its IP. Put the IP into a web browser of your choice, and append /update onto the address. You will be presented with two buttons, click the left one, choose the bin file, and click update. The bin file will be uploaded to the module and flashed by itself
An example project is included that shows how to use the driver station class. The basic steps are:
Define the Driver Station object, specify a port
CTRE.FRC.DriverStation ds = new CTRE.FRC.DriverStation(new new CTRE.HERO.Port1Definition());
Define a controller using the Driver Station object as your provider
CTRE.Controller.GameController _gamepad = new CTRE.Controller.GameController(ds, 0);
Call ds.update in your main loop
ds.update();
Treat the controller as a normal controller
float y = _gamepad.getAxis(1);
For added functionality use SendBattery, SendIP, and SendUDP methods
ds.SendIP(new byte[] { 10, 0, 33, 2 }, new byte[] { 10, 0, 33, 5 });
ds.SendBattery(12.34f);
ds.SendUDP(1234, BitConverter.UTF8.GetBytes("This is sent through UDP"));
The Module firmware currently hard codes the SSID and Password into the module. If you would like to change that, or do anything else with the firmware, you are free to. For SSID and Password it is inside the setup function under the function WiFi.softAP, use the search function to find it. In order to flash the module with the new firmware, follow these next few steps.
- Run the Binary Splitter project, and point to the bin file from the arduino compile
- Find the 11 bin files the project will spit out.
- Open the wifi esp flasher project
- Right click on the project under project view, and go into properties
- Go under the resources tab, in the upper left there is a drop down, make sure it's set to files
- Delete all the files inside the viewer
- Copy and paste the 11 bin files you found earlier into the viewer, click ok to every prompt
- Save the properties and run the program as if you were flashing the wifi module normally.
- You're good to go
The Driver station's protocol for sending data is largely hidden behind the scenes. Using a program called Wireshark I was able to find the individual datagrams and figure out what each byte meant, along with the ports the Driver station expects to use for transmitting and receiving data.
Further documentation on this topic is inside the documentation folder, including a wireshark capture of the roborio-dashboard discussion over USB.
- Driver Station to Robot (Port 57655 -> Port 1110)
- 1,2 Packet # - These two bytes, big endian, specify what number packet this is, and is used for latency control on the Driver Station
- 3 Unknown currently
- 4 Robot State - Specifies what state the robot is in, 0-2 is Teleop, Test, Auton disabled respectively, 4-6 is the enabled version
- 5,6 Unknown currently
- 7- Joystick data, length varies on type of joystick and how many are plugged in. For detailed information look at Driverstation Class
- Joystick Data
- 1 - Number of bytes for joystick
- 2,3 - Unknown currently (I think it's joystick model)
- 4 - X number of joysticks
- 5-(X+4) - Joystick axis data (signed byte) (each byte is an axis)
- (X+5) - number of buttons
- (X+6),(X+7) - bitmap of buttons
- (X+8) - Number of POV's/Hats
- (X+9),(x+10) - POV direction (unsigned byte)
- Robot to Driver Station (Port 34959 -> Port 1150)
- 1-4 Same first four bytes from Driver station, ensures packets were sent correctly
- 5-6 Battery voltage, first byte is integer voltage, second is decimal voltage
- 7 - Comms/Robot code, 0x31 for Comms and robot code, 0x10 for just Comms
- 8- Unknown currently
Below is a picture of a capture from Wireshark with the UDP packet from the computer to the RoboRIO, only one joystick connected.
The ESP's protocol to flashing an image into its flash is controlled through a ROM bootloader. This bootloader expects certain packets to come in, with details on the packet in the packet header. The overall process for flashing an image is
- Put ESP into bootloader mode by pulling GPIO0 down (Pin 3 on the HERO Gadgeteer port) and resetting the module by pulling RESET down (Pin 6 on the HERO)
- Sync baud rate of ESP to baud rate on flasher - this is done using a special packet that sends AA to the module multiple times
- Erase flash on module - Another packet is sent that specifies the amount of space needed for the flash
- Send .bin contents - Multiple packets are sent with the .bin contents inside them
- Close the bootloader - A single packet is sent with the end command to take the module out of bootloader mode.
For those wanting to create their own flashing tool, this may prove helpful
Largely based off this sheet, it details exactly what is needed in the packet header and how to go about flashing the firmmware.
Key notes:
- The module uses SLIP protocol, which means except for the header and footer, there are no 0xC0's in the packets, and so they must be replaced if they are needed.
- Checksum is created by doing XOR operation on all data bytes before the SLIP framing is done, and then that must be framed in case it is a 0xC0 or 0xDB
- Length of data is calculated before the SLIP framing
- You must make sure the other end is done talking before you can talk yourself
- Currently the amount of Flash to be erased is hard coded in this code, if you want to flash a large file you must change that byte word yourself
- I have only encounted two kinds of errors from the module, a 0x01 0x07 & 0x01 0x05
- 0x07 - Not fatal, module can still be flashed (I believe it is warning the flasher that it is currently downloading the last bin packet)
- 0x05 - Fatal, module must be reflashed (I believe it is a checksum error)