diff --git a/README.md b/README.md index 3c74eac..08652ce 100644 --- a/README.md +++ b/README.md @@ -21,7 +21,7 @@
- GP2040-CE is compatible with PC, PS3 and PS4, Nintendo Switch, Steam Deck, MiSTer and Android. + GP2040-CE is compatible with PC, PS3, PS4, PS5, Nintendo Switch, Xbox One, Steam Deck, MiSTer and Android.
## Links @@ -33,7 +33,7 @@ Full documentation can be found at [https://gp2040-ce.info](https://gp2040-ce.in ## Features - Select from 13 input modes including X-Input, Nintendo Switch, Playstation 4/5, Xbox One, D-Input, and Keyboard -- Overclocked polling rate for an average of 0.76ms of input latency in Xinput and on average 1.72 for Playstation 4/5. +- Input latency average of 0.76ms in Xinput and 0.91ms for Playstation 5. - Multiple SOCD cleaning modes - Up Priority (a.k.a. Stickless), Neutral, and Second Input Priority. - Left and Right stick emulation via D-pad inputs as well as dedicated toggle switches. - Dual direction via D-pad + LS/RS. diff --git a/development/debugging.mdx b/development/debugging.mdx new file mode 100644 index 0000000..cafd7cf --- /dev/null +++ b/development/debugging.mdx @@ -0,0 +1,68 @@ +--- +title: Debugging +# tags: +# - +pagination_next: null +pagination_prev: null +description: "Documentation on how to set up a GP2040-CE device for debugging while developing firmware features and add-ons." +--- + +# Debugging + +Debugging is essential when developing anything in the firmware because it will help you identify and fix errors in your code, ensuring that things runs correctly. It allows you to detect syntax, logic, and runtime issues that could cause your program to crash or behave unexpectedly. The RP2040 interacts with various peripherals (e.g., GPIO, I2C, SPI). Debugging helps you ensure that these interactions are working correctly, especially when dealing with timing-sensitive operations. + +By catching and resolving issues early in the development process, debugging helps you build more reliable and efficient code, saving time and preventing complex problems later on. + +## UART (TTL) to USB Serial Monitor + +To set up an RP2040 microcontroller to use UART communication over USB on GPIO 00 and 01, follow these steps: + +### Pin Configuration + +The RP2040 has two UART peripherals (UART0 and UART1). To use GPIO 00 and GPIO 01 for UART communication, you will need to leave GPIO 00 and 01 free in all GPIO pin assignments in the Board config including I2C add-ons and display, USB Host assignments or button assignments. Here's how the pins map: + +- GPIO 00: UART0 TX (Transmit) +- GPIO 01: UART0 RX (Receive) + +### Circuit Setup + +![USB - UART Connection](https://hackster.imgix.net/uploads/attachments/1284607/image_eiE7x8afXR.png?auto=compress%2Cformat&w=740&h=555&fit=max) + +- Connect USB Host to your PC to use it as a controller. +- Connect GPIO 00 (TX) to the RX pin of the USB-to-UART Serial Port Module. +- Connect GPIO 01 (RX) to the TX pin of the USB-to-UART Serial Port Module. + +Make sure the ground (GND) of the RP2040 is connected to the ground of the USB-to-UART Serial Port Module to establish a common reference. + +:::tip + +In the event that no messages appear in serial monitor, reverse the pin connections. + +::: + +### Software Configuration + +In the firmware, you will need to + +1. Add `stdio_init_all();` to the `process()` function of your feature or add-on +2. Include the `iostream` C++ library to the executable (i.e. file.cpp) +3. Use `printf()` will output to whatever serial monitor you use to read the USB serial port. + +For formatting the serial message, see https://cplusplus.com/reference/cstdio/printf/ + +### Read Messages via Serial Monitor + +There are a number of serial monitor applications that can be used to read printed messages. Examples include, but are not limited to: + +- [PuTTY](https://www.putty.org/) (Windows or Linux) +- [iTerm2](https://iterm2.com/) (MacOS) +- [picocom](https://github.com/npat-efault/picocom) (Linux) +- [Serial Monitor](https://marketplace.visualstudio.com/items?itemName=ms-vscode.vscode-serial-monitor) VS Code Extension (Windows, MacOS, Linux) +- [Windows Terminal](https://github.com/microsoft/terminal) (Windows) +- Default Terminal Application (Linux or MacOS) + +:::note Baud Rate + +By default, the baud rate is set to 115200Hz. + +::: diff --git a/development/modifying-gamepad-state.mdx b/development/modifying-gamepad-state.mdx new file mode 100644 index 0000000..0eaa3f6 --- /dev/null +++ b/development/modifying-gamepad-state.mdx @@ -0,0 +1,127 @@ +--- +title: Modifying Gamepad State +# tags: +# - +pagination_next: null +pagination_prev: null +description: Documentation on how to modify gamepad state when developing firmware features and add-ons" +--- + +# Modifying Gamepad State + +In any function looking to modify the gamepad state, use a pointer to the gamepad. This is absolutely necessary for changing any aspect of gamepad state. + +```cpp +Gamepad *gamepad = Storage::getInstance().GetGamepad(); +``` + +You can find the appropriate gamepad macros and button masks in `headers/gamepad/GamepadState.h` + +## Directional Pad + +To modify the gamepad's directional pad (DPad) state, use a bitwise OR assignment and the appropriate `GAMEPAD_MASK` enum to set the input as active. + +```cpp +Gamepad *gamepad = Storage::getInstance().GetGamepad(); +gamepad->state.dpad |= GAMEPAD_MASK_UP; +gamepad->state.dpad |= GAMEPAD_MASK_DOWN; +gamepad->state.dpad |= GAMEPAD_MASK_LEFT; +gamepad->state.dpad |= GAMEPAD_MASK_RIGHT; +``` + +:::note SOCD Cleaning for Add-ons + +All add-ons need to handle opposing simultaneous cardinal directions within the add-on since they come after standard GPIO inputs has already been SOCD cleaned. + +Consider using `runSOCDCleaner(SOCDMode mode, uint8_t dpad)` in `src/gamepad/GamepadState.cpp` to take incoming DPad inputs and then resolve them at once in the add-on's `process()` function. + +::: + +## Buttons + +To modify the gamepad's button states, use a bitwise OR assignment and the appropriate `GAMEPAD_MASK` enum to set the input as active. + +```cpp +Gamepad *gamepad = Storage::getInstance().GetGamepad(); +gamepad->state.buttons |= GAMEPAD_MASK_B1; +gamepad->state.buttons |= GAMEPAD_MASK_B2; +gamepad->state.buttons |= GAMEPAD_MASK_B3; +gamepad->state.buttons |= GAMEPAD_MASK_B4; +gamepad->state.buttons |= GAMEPAD_MASK_L1; +gamepad->state.buttons |= GAMEPAD_MASK_R1; +gamepad->state.buttons |= GAMEPAD_MASK_L2; +gamepad->state.buttons |= GAMEPAD_MASK_R2; +gamepad->state.buttons |= GAMEPAD_MASK_S1; +gamepad->state.buttons |= GAMEPAD_MASK_S2; +gamepad->state.buttons |= GAMEPAD_MASK_L3; +gamepad->state.buttons |= GAMEPAD_MASK_R3; +gamepad->state.buttons |= GAMEPAD_MASK_A1; +gamepad->state.buttons |= GAMEPAD_MASK_A2; +gamepad->state.buttons |= AUX_MASK_FUNCTION; +``` + +## Joystick + +To modify the gamepad's analog joystick state, each joystick's X- and Y-axes must be set to an unsigned integer between 0 and 65535. + +```cpp +GAMEPAD_JOYSTICK_MIN = 0 +GAMEPAD_JOYSTICK_MID = 0x7FFF +GAMEPAD_JOYSTICK_MAX = 0xFFFF +``` + +| Axis | Value Range | Joystick Position | +| :----- | :-------------- | :------------------------ | +| X-Axis | 0 - 32767 | Direction Left 100% - 0% | +| X-Axis | 32767 | Centered | +| X-Axis | 32767 - 65535 | Direction Right 0% - 100% | +| Y-Axis | 0 - 32767 | Direction Up 100% - 0% | +| Y-Axis | 32767 | Centered | +| Y-Axis | 32767 - 65535 | Direction Down 0% - 100% | + +```cpp +Gamepad *gamepad = Storage::getInstance().GetGamepad(); +gamepad->state.ly = value; +gamepad->state.lx = value; +gamepad->state.ry = value: +gamepad->state.rx = value: +``` + +:::note SOCD Cleaning for Analog Inputs + +All add-ons will need to handle opposing simultaneous cardinal directions on the analog joysticks as there are no preexisting mechanisms for resolving such things elsewhere in the firmware. + +::: + +## Triggers + +For the L2 and R2 triggers, there are two different methods for setting the values of the triggers. You can either set them digitally or using analog values. + +### Digital Triggers + +Similarly to other digital inputs, use a bitwise OR assignment and the appropriate `GAMEPAD_MASK` enum to set the input as active. + +```cpp +Gamepad *gamepad = Storage::getInstance().GetGamepad(); +gamepad->state.buttons |= GAMEPAD_MASK_L2; +gamepad->state.buttons |= GAMEPAD_MASK_R2; +``` + +### Analog Triggers + +To set the L2 and R2 triggers to analog values, first, the gamepad's flag `hasAnalogTriggers` indicating whether the gamepad has analog triggers must be set to `true`. By default, this flag is set to `false` and it is recommended that you set this flag whenever you are manipulating analog trigger values. + +Then, the appropriate axes should be set for each analog trigger. The values for analog triggers can range from 0 to 255, where 0 is fully released and 255 is fully depressed. + +```cpp +GAMEPAD_TRIGGER_MIN = 0; +GAMEPAD_TRIGGER_MID = 0x7F; +GAMEPAD_TRIGGER_MAX = 0xFF; +``` + +```cpp +Gamepad *gamepad = Storage::getInstance().GetGamepad(); +gamepad->hasAnalogTriggers = true; +gamepad->state.lt = value; +gamepad->state.rt = value; +``` diff --git a/sidebarsDevelopment.json b/sidebarsDevelopment.json index 5b9f86d..7e50bde 100644 --- a/sidebarsDevelopment.json +++ b/sidebarsDevelopment.json @@ -3,9 +3,12 @@ "contribution-guide", { "type": "category", - "label": "Firmware", + "label": "Firmware Development", "collapsed": false, - "items": ["firmware-development"] + "items": [ + "modifying-gamepad-state", + "debugging" + ] }, { "type": "category",