You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

526 lines
15 KiB

Merge ChibiOS and LUFA descriptor support (#2362) * Move lufa descriptor to protocol/usb_descriptor * Try to compile usb_descriptor on ChibiOS * Add lufa_utils for ChibiOS Lufa USB descriptors for ChibiOS * More lufa_util compatibility fixes * First compiling version of shared USB descriptor * Send the usb descriptors * Fix the CONSOLE output on ChibiOS * Add errors for unsupported interfaces * Enable support for vitual serial port USB descriptors * Implement virtual serial port for ChibiOS * Cleanup the lufa_utils Use the default lufa header files * Add raw hid support for ChibiOS This is completely untested * Enable midi compilation on ChibiOS * Move midi functionality out of lufa.c * Don't register sysex callback when not needed * ChibiOS compilation fixes * Update ChibiOS submodule * Fix the Midi USB descriptor It didn't work properly when both Midi and Virtual serial port was enabled. * Add MIDI support for ChibiOS * Fix USB descriptor strings on ChibiOS * Use serial usb driver for raw hid * Generalize the ChibiOS stream like drivers This makes the initialization much more simple and eliminates a lot of the code duplication. * Convert console output to chibios stream driver * Fixes for ChibiOS update * Update the ChibiOS contrib submodule To include the usb data toggle synchronization fixes * Fix duplicate reset enumeration on ChibiOS * Add missing include * Add number of endpoints check for ChibiOS * Enable serial USB driver on all keyboards * Add missing includes when API is enabled withot midi * Add another missing inlcude
6 years ago
Merge ChibiOS and LUFA descriptor support (#2362) * Move lufa descriptor to protocol/usb_descriptor * Try to compile usb_descriptor on ChibiOS * Add lufa_utils for ChibiOS Lufa USB descriptors for ChibiOS * More lufa_util compatibility fixes * First compiling version of shared USB descriptor * Send the usb descriptors * Fix the CONSOLE output on ChibiOS * Add errors for unsupported interfaces * Enable support for vitual serial port USB descriptors * Implement virtual serial port for ChibiOS * Cleanup the lufa_utils Use the default lufa header files * Add raw hid support for ChibiOS This is completely untested * Enable midi compilation on ChibiOS * Move midi functionality out of lufa.c * Don't register sysex callback when not needed * ChibiOS compilation fixes * Update ChibiOS submodule * Fix the Midi USB descriptor It didn't work properly when both Midi and Virtual serial port was enabled. * Add MIDI support for ChibiOS * Fix USB descriptor strings on ChibiOS * Use serial usb driver for raw hid * Generalize the ChibiOS stream like drivers This makes the initialization much more simple and eliminates a lot of the code duplication. * Convert console output to chibios stream driver * Fixes for ChibiOS update * Update the ChibiOS contrib submodule To include the usb data toggle synchronization fixes * Fix duplicate reset enumeration on ChibiOS * Add missing include * Add number of endpoints check for ChibiOS * Enable serial USB driver on all keyboards * Add missing includes when API is enabled withot midi * Add another missing inlcude
6 years ago
Velocikey: Match RGB animation speed to typing speed (#3754) * Draft commit of typing speed RGB control * More information in the readme * Support all RGB animation modes (Fixes #1) * Added support for all RGB light modes to use typing speed Except christmas lights because that is seizure-inducing at high speeds! * Introduced a value range specific to each RGB mode Because some modes are a little too much when running at full speed! * Update readme.md * Update readme.md * Re-arrange typing_speed definitions (Fixes #5) (#6) * Re-arrange variable definitions to avoid including quantum.h from rgblight.c * Fix a compilation error when trying to run make test:all * Tweaks to the typing speed decay rate * Renamed to momentum; moved implementation into dedicated files * Groundwork for toggling momentum on/off (currently always on) * Add EEPROM toggle for momentum-matching * Moved momentum out of RGBLIGHT_ENABLE toggles so it's more generic * Move momentum decay task out of rgblight_task() * Fix missing momentum.h in lufa.c * Experimental LED support (untested) * Draft commit of typing speed RGB control * More information in the readme * Support all RGB animation modes (Fixes #1) * Added support for all RGB light modes to use typing speed Except christmas lights because that is seizure-inducing at high speeds! * Introduced a value range specific to each RGB mode Because some modes are a little too much when running at full speed! * Update readme.md * Update readme.md * Re-arrange typing_speed definitions (Fixes #5) (#6) * Re-arrange variable definitions to avoid including quantum.h from rgblight.c * Fix a compilation error when trying to run make test:all * Tweaks to the typing speed decay rate * Renamed to momentum; moved implementation into dedicated files * Groundwork for toggling momentum on/off (currently always on) * Add EEPROM toggle for momentum-matching * Moved momentum out of RGBLIGHT_ENABLE toggles so it's more generic * Move momentum decay task out of rgblight_task() * Fix missing momentum.h in lufa.c * Added documentation * Renamed feature to velocikey * Reverted readme to original state * Correct the readme title * Updated feature name in the docs * Update EECONFIG name * Add compile-time toggles for velocikey * Update feature documentation * Revert "Merge branch 'led-support' into master" This reverts commit e123ff5febf61639b9a9020748e1c2e2313460ff, reversing changes made to df111a55b9d4929182e16108b1c0ead15b16df97. * Move velocikey EECONFIG definition to depend on VELOCIKEY_ENABLE * Rename decay_task function to decelerate * Apply suggestions from code review Co-Authored-By: chrislewisdev <chris@chrislewisdev.com> * Re-order eeconfig definitions * Apply coding conventions * Apply #ifdef check in lufa.c * Refactored interval time checks into one functionc * Small rename * Fix unused function error for layouts not using all rgb effects * Only update EEPROM if Velocikey is enabled * Incorporate code review feedback * Small adjustment to top-end decay rate * Add Velocikey documentation to table of contents * Bring tetris:default keymap size down by disabling audio
5 years ago
Velocikey: Match RGB animation speed to typing speed (#3754) * Draft commit of typing speed RGB control * More information in the readme * Support all RGB animation modes (Fixes #1) * Added support for all RGB light modes to use typing speed Except christmas lights because that is seizure-inducing at high speeds! * Introduced a value range specific to each RGB mode Because some modes are a little too much when running at full speed! * Update readme.md * Update readme.md * Re-arrange typing_speed definitions (Fixes #5) (#6) * Re-arrange variable definitions to avoid including quantum.h from rgblight.c * Fix a compilation error when trying to run make test:all * Tweaks to the typing speed decay rate * Renamed to momentum; moved implementation into dedicated files * Groundwork for toggling momentum on/off (currently always on) * Add EEPROM toggle for momentum-matching * Moved momentum out of RGBLIGHT_ENABLE toggles so it's more generic * Move momentum decay task out of rgblight_task() * Fix missing momentum.h in lufa.c * Experimental LED support (untested) * Draft commit of typing speed RGB control * More information in the readme * Support all RGB animation modes (Fixes #1) * Added support for all RGB light modes to use typing speed Except christmas lights because that is seizure-inducing at high speeds! * Introduced a value range specific to each RGB mode Because some modes are a little too much when running at full speed! * Update readme.md * Update readme.md * Re-arrange typing_speed definitions (Fixes #5) (#6) * Re-arrange variable definitions to avoid including quantum.h from rgblight.c * Fix a compilation error when trying to run make test:all * Tweaks to the typing speed decay rate * Renamed to momentum; moved implementation into dedicated files * Groundwork for toggling momentum on/off (currently always on) * Add EEPROM toggle for momentum-matching * Moved momentum out of RGBLIGHT_ENABLE toggles so it's more generic * Move momentum decay task out of rgblight_task() * Fix missing momentum.h in lufa.c * Added documentation * Renamed feature to velocikey * Reverted readme to original state * Correct the readme title * Updated feature name in the docs * Update EECONFIG name * Add compile-time toggles for velocikey * Update feature documentation * Revert "Merge branch 'led-support' into master" This reverts commit e123ff5febf61639b9a9020748e1c2e2313460ff, reversing changes made to df111a55b9d4929182e16108b1c0ead15b16df97. * Move velocikey EECONFIG definition to depend on VELOCIKEY_ENABLE * Rename decay_task function to decelerate * Apply suggestions from code review Co-Authored-By: chrislewisdev <chris@chrislewisdev.com> * Re-order eeconfig definitions * Apply coding conventions * Apply #ifdef check in lufa.c * Refactored interval time checks into one functionc * Small rename * Fix unused function error for layouts not using all rgb effects * Only update EEPROM if Velocikey is enabled * Incorporate code review feedback * Small adjustment to top-end decay rate * Add Velocikey documentation to table of contents * Bring tetris:default keymap size down by disabling audio
5 years ago
Backlighting for JJ40 and underglow initialisation code (#2260) * Cleanup Mechmini keymap. Once the custom RGB function is defined, there is no need to manually handle RGB code. * Change default to KEYMAP_MIT, not KEYMAP_OFFSET * Add custom RGB code for JJ40 * Reset Mechmini advertised power draw to 500. Will have to test actual maximum power draw later. * RGB working on JJ40. * Fix: saturation increase/decrease flipped * Add new directory for my custom keymap with RGB keycodes * Swap LAlt and LGUI * Update JJ40 max power draw with measured value * Update: fun40 rules.mk to enable underglow; earlier failed Travis CI * Fix: init RGB LEDs on boot. Also added HHKB-like keymap for XD60. * Super rudimentary backlight test, init RGB LEDs on boot * Backlighting works - stays on for now * Toggling working * Now can override backlight.c functions. Problem was functions in backlight.c weren't called before due to a lack of matrix_scan_quantum() in matrix.c * Timers not working * Delete global.h * Cleanup * Compiles * Good sign: LEDs stop working again * Handle timer1 overflow * Progress: fix: forgot to init * Backlighting fully working now except breathing. * Revert keymap to original keycodes * Update XD60 keymap README * Update JJ40 keymap with backlight toggles * Breathing working just fine. * Update references * Add backlight_set() call * Cleanup code to disable backlight * Fix: does not compile * Fix: missing call to rgblight_task. * Testing with BACKLIGHT_BREATHING * Cleanup * Cleanup comments * More commenting cleanup. * Do not enable BACKLIGHT_BREATHING by default
6 years ago
Process combos earlier & overlapping combos (#8591) * Combo processing improvements. Now it is possible to use ModTap and LayerTap keys as part of combos. Overlapping combos also don't trigger all the combos, just exactly the one that you press. New settings: - COMBO_MUST_HOLD_MODS - COMBO_MOD_TERM - COMBO_TERM_PER_COMBO - COMBO_MUST_HOLD_PER_COMBO - COMBO_STRICT_TIMER - COMBO_NO_TIMER * Remove the size flags from combo_t struct boolean members. This in the end actually saves space as the members are accessed so many times. The amount of operations needed to access the bits uses more memory than setting the size saves. * Fix `process_combo_key_release` not called correctly with tap-only combos * Fix not passing a pointer when NO_ACTION_TAPPING is defined. * Docs for `COMBO_ONLY_FROM_LAYER` * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update quantum/process_keycode/process_combo.c Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Add `EXTRA_SHORT_COMBOS` option. Stuff combo's `disabled` and `active` flags into `state`. Possibly can save some space. * Add more examples and clarify things with dict management system. - Simple examples now has a combo that has modifiers included. - The slightly more advanced examples now are actually more advanced instead of just `tap_code16(<modded-keycode>)`. - Added a note that `COMBO_ACTION`s are not needed anymore as you can just use custom keycodes. - Added a note that the `g/keymap_combo.h` macros use the `process_combo_event` function and that it is not usable in one's keymap afterwards. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Change "the" combo action example to "email" example. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Fix sneaky infinite loop with `combo_disable()` No need to call `dump_key_buffer` when disabling combos because the buffer is either being dumped if a combo-key was pressed, or the buffer is empty if a non-combo-key is pressed. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> Co-authored-by: Drashna Jaelre <drashna@live.com>
2 years ago
Process combos earlier & overlapping combos (#8591) * Combo processing improvements. Now it is possible to use ModTap and LayerTap keys as part of combos. Overlapping combos also don't trigger all the combos, just exactly the one that you press. New settings: - COMBO_MUST_HOLD_MODS - COMBO_MOD_TERM - COMBO_TERM_PER_COMBO - COMBO_MUST_HOLD_PER_COMBO - COMBO_STRICT_TIMER - COMBO_NO_TIMER * Remove the size flags from combo_t struct boolean members. This in the end actually saves space as the members are accessed so many times. The amount of operations needed to access the bits uses more memory than setting the size saves. * Fix `process_combo_key_release` not called correctly with tap-only combos * Fix not passing a pointer when NO_ACTION_TAPPING is defined. * Docs for `COMBO_ONLY_FROM_LAYER` * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update quantum/process_keycode/process_combo.c Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Add `EXTRA_SHORT_COMBOS` option. Stuff combo's `disabled` and `active` flags into `state`. Possibly can save some space. * Add more examples and clarify things with dict management system. - Simple examples now has a combo that has modifiers included. - The slightly more advanced examples now are actually more advanced instead of just `tap_code16(<modded-keycode>)`. - Added a note that `COMBO_ACTION`s are not needed anymore as you can just use custom keycodes. - Added a note that the `g/keymap_combo.h` macros use the `process_combo_event` function and that it is not usable in one's keymap afterwards. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Change "the" combo action example to "email" example. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Fix sneaky infinite loop with `combo_disable()` No need to call `dump_key_buffer` when disabling combos because the buffer is either being dumped if a combo-key was pressed, or the buffer is empty if a non-combo-key is pressed. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> Co-authored-by: Drashna Jaelre <drashna@live.com>
2 years ago
Process combos earlier & overlapping combos (#8591) * Combo processing improvements. Now it is possible to use ModTap and LayerTap keys as part of combos. Overlapping combos also don't trigger all the combos, just exactly the one that you press. New settings: - COMBO_MUST_HOLD_MODS - COMBO_MOD_TERM - COMBO_TERM_PER_COMBO - COMBO_MUST_HOLD_PER_COMBO - COMBO_STRICT_TIMER - COMBO_NO_TIMER * Remove the size flags from combo_t struct boolean members. This in the end actually saves space as the members are accessed so many times. The amount of operations needed to access the bits uses more memory than setting the size saves. * Fix `process_combo_key_release` not called correctly with tap-only combos * Fix not passing a pointer when NO_ACTION_TAPPING is defined. * Docs for `COMBO_ONLY_FROM_LAYER` * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update quantum/process_keycode/process_combo.c Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Add `EXTRA_SHORT_COMBOS` option. Stuff combo's `disabled` and `active` flags into `state`. Possibly can save some space. * Add more examples and clarify things with dict management system. - Simple examples now has a combo that has modifiers included. - The slightly more advanced examples now are actually more advanced instead of just `tap_code16(<modded-keycode>)`. - Added a note that `COMBO_ACTION`s are not needed anymore as you can just use custom keycodes. - Added a note that the `g/keymap_combo.h` macros use the `process_combo_event` function and that it is not usable in one's keymap afterwards. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Change "the" combo action example to "email" example. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Fix sneaky infinite loop with `combo_disable()` No need to call `dump_key_buffer` when disabling combos because the buffer is either being dumped if a combo-key was pressed, or the buffer is empty if a non-combo-key is pressed. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> Co-authored-by: Drashna Jaelre <drashna@live.com>
2 years ago
Process combos earlier & overlapping combos (#8591) * Combo processing improvements. Now it is possible to use ModTap and LayerTap keys as part of combos. Overlapping combos also don't trigger all the combos, just exactly the one that you press. New settings: - COMBO_MUST_HOLD_MODS - COMBO_MOD_TERM - COMBO_TERM_PER_COMBO - COMBO_MUST_HOLD_PER_COMBO - COMBO_STRICT_TIMER - COMBO_NO_TIMER * Remove the size flags from combo_t struct boolean members. This in the end actually saves space as the members are accessed so many times. The amount of operations needed to access the bits uses more memory than setting the size saves. * Fix `process_combo_key_release` not called correctly with tap-only combos * Fix not passing a pointer when NO_ACTION_TAPPING is defined. * Docs for `COMBO_ONLY_FROM_LAYER` * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update quantum/process_keycode/process_combo.c Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Add `EXTRA_SHORT_COMBOS` option. Stuff combo's `disabled` and `active` flags into `state`. Possibly can save some space. * Add more examples and clarify things with dict management system. - Simple examples now has a combo that has modifiers included. - The slightly more advanced examples now are actually more advanced instead of just `tap_code16(<modded-keycode>)`. - Added a note that `COMBO_ACTION`s are not needed anymore as you can just use custom keycodes. - Added a note that the `g/keymap_combo.h` macros use the `process_combo_event` function and that it is not usable in one's keymap afterwards. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Change "the" combo action example to "email" example. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Fix sneaky infinite loop with `combo_disable()` No need to call `dump_key_buffer` when disabling combos because the buffer is either being dumped if a combo-key was pressed, or the buffer is empty if a non-combo-key is pressed. * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> * Update docs/feature_combo.md Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> Co-authored-by: precondition <57645186+precondition@users.noreply.github.com> Co-authored-by: Drashna Jaelre <drashna@live.com>
2 years ago
Velocikey: Match RGB animation speed to typing speed (#3754) * Draft commit of typing speed RGB control * More information in the readme * Support all RGB animation modes (Fixes #1) * Added support for all RGB light modes to use typing speed Except christmas lights because that is seizure-inducing at high speeds! * Introduced a value range specific to each RGB mode Because some modes are a little too much when running at full speed! * Update readme.md * Update readme.md * Re-arrange typing_speed definitions (Fixes #5) (#6) * Re-arrange variable definitions to avoid including quantum.h from rgblight.c * Fix a compilation error when trying to run make test:all * Tweaks to the typing speed decay rate * Renamed to momentum; moved implementation into dedicated files * Groundwork for toggling momentum on/off (currently always on) * Add EEPROM toggle for momentum-matching * Moved momentum out of RGBLIGHT_ENABLE toggles so it's more generic * Move momentum decay task out of rgblight_task() * Fix missing momentum.h in lufa.c * Experimental LED support (untested) * Draft commit of typing speed RGB control * More information in the readme * Support all RGB animation modes (Fixes #1) * Added support for all RGB light modes to use typing speed Except christmas lights because that is seizure-inducing at high speeds! * Introduced a value range specific to each RGB mode Because some modes are a little too much when running at full speed! * Update readme.md * Update readme.md * Re-arrange typing_speed definitions (Fixes #5) (#6) * Re-arrange variable definitions to avoid including quantum.h from rgblight.c * Fix a compilation error when trying to run make test:all * Tweaks to the typing speed decay rate * Renamed to momentum; moved implementation into dedicated files * Groundwork for toggling momentum on/off (currently always on) * Add EEPROM toggle for momentum-matching * Moved momentum out of RGBLIGHT_ENABLE toggles so it's more generic * Move momentum decay task out of rgblight_task() * Fix missing momentum.h in lufa.c * Added documentation * Renamed feature to velocikey * Reverted readme to original state * Correct the readme title * Updated feature name in the docs * Update EECONFIG name * Add compile-time toggles for velocikey * Update feature documentation * Revert "Merge branch 'led-support' into master" This reverts commit e123ff5febf61639b9a9020748e1c2e2313460ff, reversing changes made to df111a55b9d4929182e16108b1c0ead15b16df97. * Move velocikey EECONFIG definition to depend on VELOCIKEY_ENABLE * Rename decay_task function to decelerate * Apply suggestions from code review Co-Authored-By: chrislewisdev <chris@chrislewisdev.com> * Re-order eeconfig definitions * Apply coding conventions * Apply #ifdef check in lufa.c * Refactored interval time checks into one functionc * Small rename * Fix unused function error for layouts not using all rgb effects * Only update EEPROM if Velocikey is enabled * Incorporate code review feedback * Small adjustment to top-end decay rate * Add Velocikey documentation to table of contents * Bring tetris:default keymap size down by disabling audio
5 years ago
2020 November 28 Breaking Changes Update (#11053) * Branch point for 2020 November 28 Breaking Change * Remove matrix_col_t to allow MATRIX_ROWS > 32 (#10183) * Add support for soft serial to ATmega32U2 (#10204) * Change MIDI velocity implementation to allow direct control of velocity value (#9940) * Add ability to build a subset of all keyboards based on platform. * Actually use eeprom_driver_init(). * Make bootloader_jump weak for ChibiOS. (#10417) * Joystick 16-bit support (#10439) * Per-encoder resolutions (#10259) * Share button state from mousekey to pointing_device (#10179) * Add hotfix for chibios keyboards not wake (#10088) * Add advanced/efficient RGB Matrix Indicators (#8564) * Naming change. * Support for STM32 GPIOF,G,H,I,J,K (#10206) * Add milc as a dependency and remove the installed milc (#10563) * ChibiOS upgrade: early init conversions (#10214) * ChibiOS upgrade: configuration file migrator (#9952) * Haptic and solenoid cleanup (#9700) * XD75 cleanup (#10524) * OLED display update interval support (#10388) * Add definition based on currently-selected serial driver. (#10716) * New feature: Retro Tapping per key (#10622) * Allow for modification of output RGB values when using rgblight/rgb_matrix. (#10638) * Add housekeeping task callbacks so that keyboards/keymaps are capable of executing code for each main loop iteration. (#10530) * Rescale both ChibiOS and AVR backlighting. * Reduce Helix keyboard build variation (#8669) * Minor change to behavior allowing display updates to continue between task ticks (#10750) * Some GPIO manipulations in matrix.c change to atomic. (#10491) * qmk cformat (#10767) * [Keyboard] Update the Speedo firmware for v3.0 (#10657) * Maartenwut/Maarten namechange to evyd13/Evy (#10274) * [quantum] combine repeated lines of code (#10837) * Add step sequencer feature (#9703) * aeboards/ext65 refactor (#10820) * Refactor xelus/dawn60 for Rev2 later (#10584) * add DEBUG_MATRIX_SCAN_RATE_ENABLE to common_features.mk (#10824) * [Core] Added `add_oneshot_mods` & `del_oneshot_mods` (#10549) * update chibios os usb for the otg driver (#8893) * Remove HD44780 References, Part 4 (#10735) * [Keyboard] Add Valor FRL TKL (+refactor) (#10512) * Fix cursor position bug in oled_write_raw functions (#10800) * Fixup version.h writing when using SKIP_VERSION=yes (#10972) * Allow for certain code in the codebase assuming length of string. (#10974) * Add AT90USB support for serial.c (#10706) * Auto shift: support repeats and early registration (#9826) * Rename ledmatrix.h to match .c file (#7949) * Split RGB_MATRIX_ENABLE into _ENABLE and _DRIVER (#10231) * Split LED_MATRIX_ENABLE into _ENABLE and _DRIVER (#10840) * Merge point for 2020 Nov 28 Breaking Change
3 years ago
Hid joystick interface (#4226) * add support for hid gamepad interface add documentation for HID joystick Add joystick_task to read analog axes values even when no key is pressed or release. update doc Update docs/feature_joystick.md Manage pin setup and read to maintain matrix scan after analog read * Incorporates patches and changes to HID reporting There are some patches provided by @a-chol incorporated on this commit, and also some changes I made to the HID Report structure. The most interesting is the one dealing with number of buttons: Linux doesn't seem to care, but Windows requires the HID structure to be byte aligned (that's in the spec). So if one declares 8/16/32... buttons they should not have any issues, but this is what happens when you have 9 buttons: ``` bits |0|1|2|3|4|5|6|7| |*|*|*|*|*|*|*|*| axis 0 (report size 8) |*|*|*|*|*|*|*|*| ... |*|*|*|*|*|*|*|*| |*|*|*|*|*|*|*|*| |*|*|*|*|*|*|*|*| |*|*|*|*|*|*|*|*| |*|*|*|*|*|*|*|*| axis 6 |*|*|*|*|*|*|*|*| first 8 buttons (report size 1) |*| | | | | | | | last of 9 buttons, not aligned ``` So for that I added a conditonal that will add a number of reports with size 1 to make sure it aligns to the next multiple of 8. Those reports send dummy inputs that don't do anything aside from aligning the data. Tested on Linux, Windows 10 and Street Fighter (where the joystick is recognized as direct-input) * Add save and restore of each pin used in reading joystick (AVR). Allow output pin to be JS_VIRTUAL_AXIS if the axis is connected to Vcc instead of an output pin from the MCU. Fix joystick report id Fix broken v-usb hid joystick interface. Make it more resilient to unusual settings (none multiple of eight button count, 0 buttons or 0 axes) Correct adc reading for multiple axes. Piecewise range conversion for uncentered raw value range. Input, output and ground pin configuration per axis. Documentation fixes * Fix port addressing for joystick analog read * The other required set of changes As per the PR, the changes still holding it up. Add onekey for testing. Fix ARM builds. Fix device descriptor when either axes or buttons is zero. Add compile-time check for at least one axis or button. Move definition to try to fix conflict. PR review comments. qmk cformat * avoid float functions to compute range mapping for axis adc reading * Remove V-USB support for now. Updated docs accordingly. * Update tmk_core/protocol/lufa/lufa.c Co-Authored-By: Ryan <fauxpark@gmail.com> * Update tmk_core/protocol/usb_descriptor.c Co-Authored-By: Ryan <fauxpark@gmail.com> * Update tmk_core/protocol/usb_descriptor.c Co-Authored-By: Ryan <fauxpark@gmail.com> * Update tmk_core/protocol/usb_descriptor.c Co-Authored-By: Ryan <fauxpark@gmail.com> * Add support for joystick adc reading for stm32 MCUs. Fix joystick hid report sending for chibios * Fix HID joystick report sending for ChibiOS. Add one analog axis to the onekey:joystick keymap. Fix pin state save and restore during joystick analog read for STM32 MCUs. * Update tmk_core/protocol/chibios/usb_main.c Co-Authored-By: Ryan <fauxpark@gmail.com> * Update tmk_core/protocol/lufa/lufa.c Co-Authored-By: Ryan <fauxpark@gmail.com> * Add missing mcuconf.h and halconf.h to onekey:joystick keymap. Add suggested fixes from PR. * Switch saveState and restoreState signature to use pin_t type. onekey:joystick : add a second axis, virtual and programmatically animated. * Update docs/feature_joystick.md Co-Authored-By: Ryan <fauxpark@gmail.com> * Update docs/feature_joystick.md Co-Authored-By: Ryan <fauxpark@gmail.com> * Add PR corrections * Remove halconf.h and mcuconf.h from onekey keymaps * Change ADC_PIN to A0 Co-authored-by: achol <allecooll@hotmail.com> Co-authored-by: José Júnior <jose.junior@gmail.com> Co-authored-by: a-chol <achol@notamail.com> Co-authored-by: Nick Brassel <nick@tzarc.org> Co-authored-by: Ryan <fauxpark@gmail.com>
3 years ago
  1. /* Copyright 2016-2017 Jack Humbert
  2. *
  3. * This program is free software: you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License as published by
  5. * the Free Software Foundation, either version 2 of the License, or
  6. * (at your option) any later version.
  7. *
  8. * This program is distributed in the hope that it will be useful,
  9. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. * GNU General Public License for more details.
  12. *
  13. * You should have received a copy of the GNU General Public License
  14. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. #include "quantum.h"
  17. #ifdef BLUETOOTH_ENABLE
  18. # include "outputselect.h"
  19. #endif
  20. #ifdef BACKLIGHT_ENABLE
  21. # include "backlight.h"
  22. #endif
  23. #ifdef MIDI_ENABLE
  24. # include "process_midi.h"
  25. #endif
  26. #ifdef VELOCIKEY_ENABLE
  27. # include "velocikey.h"
  28. #endif
  29. #ifdef HAPTIC_ENABLE
  30. # include "haptic.h"
  31. #endif
  32. #ifdef AUDIO_ENABLE
  33. # ifndef GOODBYE_SONG
  34. # define GOODBYE_SONG SONG(GOODBYE_SOUND)
  35. # endif
  36. float goodbye_song[][2] = GOODBYE_SONG;
  37. # ifdef DEFAULT_LAYER_SONGS
  38. float default_layer_songs[][16][2] = DEFAULT_LAYER_SONGS;
  39. # endif
  40. #endif
  41. uint8_t extract_mod_bits(uint16_t code) {
  42. switch (code) {
  43. case QK_MODS ... QK_MODS_MAX:
  44. break;
  45. default:
  46. return 0;
  47. }
  48. uint8_t mods_to_send = 0;
  49. if (code & QK_RMODS_MIN) { // Right mod flag is set
  50. if (code & QK_LCTL) mods_to_send |= MOD_BIT(KC_RIGHT_CTRL);
  51. if (code & QK_LSFT) mods_to_send |= MOD_BIT(KC_RIGHT_SHIFT);
  52. if (code & QK_LALT) mods_to_send |= MOD_BIT(KC_RIGHT_ALT);
  53. if (code & QK_LGUI) mods_to_send |= MOD_BIT(KC_RIGHT_GUI);
  54. } else {
  55. if (code & QK_LCTL) mods_to_send |= MOD_BIT(KC_LEFT_CTRL);
  56. if (code & QK_LSFT) mods_to_send |= MOD_BIT(KC_LEFT_SHIFT);
  57. if (code & QK_LALT) mods_to_send |= MOD_BIT(KC_LEFT_ALT);
  58. if (code & QK_LGUI) mods_to_send |= MOD_BIT(KC_LEFT_GUI);
  59. }
  60. return mods_to_send;
  61. }
  62. void do_code16(uint16_t code, void (*f)(uint8_t)) {
  63. f(extract_mod_bits(code));
  64. }
  65. __attribute__((weak)) void register_code16(uint16_t code) {
  66. if (IS_MOD(code) || code == KC_NO) {
  67. do_code16(code, register_mods);
  68. } else {
  69. do_code16(code, register_weak_mods);
  70. }
  71. register_code(code);
  72. }
  73. __attribute__((weak)) void unregister_code16(uint16_t code) {
  74. unregister_code(code);
  75. if (IS_MOD(code) || code == KC_NO) {
  76. do_code16(code, unregister_mods);
  77. } else {
  78. do_code16(code, unregister_weak_mods);
  79. }
  80. }
  81. __attribute__((weak)) void tap_code16(uint16_t code) {
  82. register_code16(code);
  83. if (code == KC_CAPS_LOCK) {
  84. wait_ms(TAP_HOLD_CAPS_DELAY);
  85. } else if (TAP_CODE_DELAY > 0) {
  86. wait_ms(TAP_CODE_DELAY);
  87. }
  88. unregister_code16(code);
  89. }
  90. __attribute__((weak)) bool process_action_kb(keyrecord_t *record) {
  91. return true;
  92. }
  93. __attribute__((weak)) bool process_record_kb(uint16_t keycode, keyrecord_t *record) {
  94. return process_record_user(keycode, record);
  95. }
  96. __attribute__((weak)) bool process_record_user(uint16_t keycode, keyrecord_t *record) {
  97. return true;
  98. }
  99. __attribute__((weak)) void post_process_record_kb(uint16_t keycode, keyrecord_t *record) {
  100. post_process_record_user(keycode, record);
  101. }
  102. __attribute__((weak)) void post_process_record_user(uint16_t keycode, keyrecord_t *record) {}
  103. void reset_keyboard(void) {
  104. clear_keyboard();
  105. #if defined(MIDI_ENABLE) && defined(MIDI_BASIC)
  106. process_midi_all_notes_off();
  107. #endif
  108. #ifdef AUDIO_ENABLE
  109. # ifndef NO_MUSIC_MODE
  110. music_all_notes_off();
  111. # endif
  112. uint16_t timer_start = timer_read();
  113. PLAY_SONG(goodbye_song);
  114. shutdown_user();
  115. while (timer_elapsed(timer_start) < 250)
  116. wait_ms(1);
  117. stop_all_notes();
  118. #else
  119. shutdown_user();
  120. wait_ms(250);
  121. #endif
  122. #ifdef HAPTIC_ENABLE
  123. haptic_shutdown();
  124. #endif
  125. bootloader_jump();
  126. }
  127. /* Convert record into usable keycode via the contained event. */
  128. uint16_t get_record_keycode(keyrecord_t *record, bool update_layer_cache) {
  129. #ifdef COMBO_ENABLE
  130. if (record->keycode) {
  131. return record->keycode;
  132. }
  133. #endif
  134. return get_event_keycode(record->event, update_layer_cache);
  135. }
  136. /* Convert event into usable keycode. Checks the layer cache to ensure that it
  137. * retains the correct keycode after a layer change, if the key is still pressed.
  138. * "update_layer_cache" is to ensure that it only updates the layer cache when
  139. * appropriate, otherwise, it will update it and cause layer tap (and other keys)
  140. * from triggering properly.
  141. */
  142. uint16_t get_event_keycode(keyevent_t event, bool update_layer_cache) {
  143. #if !defined(NO_ACTION_LAYER) && !defined(STRICT_LAYER_RELEASE)
  144. /* TODO: Use store_or_get_action() or a similar function. */
  145. if (!disable_action_cache) {
  146. uint8_t layer;
  147. if (event.pressed && update_layer_cache) {
  148. layer = layer_switch_get_layer(event.key);
  149. update_source_layers_cache(event.key, layer);
  150. } else {
  151. layer = read_source_layers_cache(event.key);
  152. }
  153. return keymap_key_to_keycode(layer, event.key);
  154. } else
  155. #endif
  156. return keymap_key_to_keycode(layer_switch_get_layer(event.key), event.key);
  157. }
  158. /* Get keycode, and then process pre tapping functionality */
  159. bool pre_process_record_quantum(keyrecord_t *record) {
  160. if (!(
  161. #ifdef COMBO_ENABLE
  162. process_combo(get_record_keycode(record, true), record) &&
  163. #endif
  164. true)) {
  165. return false;
  166. }
  167. return true; // continue processing
  168. }
  169. /* Get keycode, and then call keyboard function */
  170. void post_process_record_quantum(keyrecord_t *record) {
  171. uint16_t keycode = get_record_keycode(record, false);
  172. post_process_record_kb(keycode, record);
  173. }
  174. /* Core keycode function, hands off handling to other functions,
  175. then processes internal quantum keycodes, and then processes
  176. ACTIONs. */
  177. bool process_record_quantum(keyrecord_t *record) {
  178. uint16_t keycode = get_record_keycode(record, true);
  179. // This is how you use actions here
  180. // if (keycode == KC_LEAD) {
  181. // action_t action;
  182. // action.code = ACTION_DEFAULT_LAYER_SET(0);
  183. // process_action(record, action);
  184. // return false;
  185. // }
  186. #ifdef VELOCIKEY_ENABLE
  187. if (velocikey_enabled() && record->event.pressed) {
  188. velocikey_accelerate();
  189. }
  190. #endif
  191. #ifdef WPM_ENABLE
  192. if (record->event.pressed) {
  193. update_wpm(keycode);
  194. }
  195. #endif
  196. #ifdef TAP_DANCE_ENABLE
  197. preprocess_tap_dance(keycode, record);
  198. #endif
  199. if (!(
  200. #if defined(KEY_LOCK_ENABLE)
  201. // Must run first to be able to mask key_up events.
  202. process_key_lock(&keycode, record) &&
  203. #endif
  204. #if defined(DYNAMIC_MACRO_ENABLE) && !defined(DYNAMIC_MACRO_USER_CALL)
  205. // Must run asap to ensure all keypresses are recorded.
  206. process_dynamic_macro(keycode, record) &&
  207. #endif
  208. #if defined(AUDIO_ENABLE) && defined(AUDIO_CLICKY)
  209. process_clicky(keycode, record) &&
  210. #endif
  211. #ifdef HAPTIC_ENABLE
  212. process_haptic(keycode, record) &&
  213. #endif
  214. #if defined(VIA_ENABLE)
  215. process_record_via(keycode, record) &&
  216. #endif
  217. process_record_kb(keycode, record) &&
  218. #if defined(SEQUENCER_ENABLE)
  219. process_sequencer(keycode, record) &&
  220. #endif
  221. #if defined(MIDI_ENABLE) && defined(MIDI_ADVANCED)
  222. process_midi(keycode, record) &&
  223. #endif
  224. #ifdef AUDIO_ENABLE
  225. process_audio(keycode, record) &&
  226. #endif
  227. #if defined(BACKLIGHT_ENABLE) || defined(LED_MATRIX_ENABLE)
  228. process_backlight(keycode, record) &&
  229. #endif
  230. #ifdef STENO_ENABLE
  231. process_steno(keycode, record) &&
  232. #endif
  233. #if (defined(AUDIO_ENABLE) || (defined(MIDI_ENABLE) && defined(MIDI_BASIC))) && !defined(NO_MUSIC_MODE)
  234. process_music(keycode, record) &&
  235. #endif
  236. #ifdef KEY_OVERRIDE_ENABLE
  237. process_key_override(keycode, record) &&
  238. #endif
  239. #ifdef TAP_DANCE_ENABLE
  240. process_tap_dance(keycode, record) &&
  241. #endif
  242. #if defined(UNICODE_COMMON_ENABLE)
  243. process_unicode_common(keycode, record) &&
  244. #endif
  245. #ifdef LEADER_ENABLE
  246. process_leader(keycode, record) &&
  247. #endif
  248. #ifdef PRINTING_ENABLE
  249. process_printer(keycode, record) &&
  250. #endif
  251. #ifdef AUTO_SHIFT_ENABLE
  252. process_auto_shift(keycode, record) &&
  253. #endif
  254. #ifdef DYNAMIC_TAPPING_TERM_ENABLE
  255. process_dynamic_tapping_term(keycode, record) &&
  256. #endif
  257. #ifdef TERMINAL_ENABLE
  258. process_terminal(keycode, record) &&
  259. #endif
  260. #ifdef SPACE_CADET_ENABLE
  261. process_space_cadet(keycode, record) &&
  262. #endif
  263. #ifdef MAGIC_KEYCODE_ENABLE
  264. process_magic(keycode, record) &&
  265. #endif
  266. #ifdef GRAVE_ESC_ENABLE
  267. process_grave_esc(keycode, record) &&
  268. #endif
  269. #if defined(RGBLIGHT_ENABLE) || defined(RGB_MATRIX_ENABLE)
  270. process_rgb(keycode, record) &&
  271. #endif
  272. #ifdef JOYSTICK_ENABLE
  273. process_joystick(keycode, record) &&
  274. #endif
  275. #ifdef PROGRAMMABLE_BUTTON_ENABLE
  276. process_programmable_button(keycode, record) &&
  277. #endif
  278. true)) {
  279. return false;
  280. }
  281. if (record->event.pressed) {
  282. switch (keycode) {
  283. #ifndef NO_RESET
  284. case QK_BOOTLOADER:
  285. reset_keyboard();
  286. return false;
  287. #endif
  288. #ifndef NO_DEBUG
  289. case QK_DEBUG_TOGGLE:
  290. debug_enable ^= 1;
  291. if (debug_enable) {
  292. print("DEBUG: enabled.\n");
  293. } else {
  294. print("DEBUG: disabled.\n");
  295. }
  296. #endif
  297. return false;
  298. case QK_CLEAR_EEPROM:
  299. eeconfig_init();
  300. return false;
  301. #ifdef VELOCIKEY_ENABLE
  302. case VLK_TOG:
  303. velocikey_toggle();
  304. return false;
  305. #endif
  306. #ifdef BLUETOOTH_ENABLE
  307. case OUT_AUTO:
  308. set_output(OUTPUT_AUTO);
  309. return false;
  310. case OUT_USB:
  311. set_output(OUTPUT_USB);
  312. return false;
  313. case OUT_BT:
  314. set_output(OUTPUT_BLUETOOTH);
  315. return false;
  316. #endif
  317. #ifndef NO_ACTION_ONESHOT
  318. case ONESHOT_TOGGLE:
  319. oneshot_toggle();
  320. break;
  321. case ONESHOT_ENABLE:
  322. oneshot_enable();
  323. break;
  324. case ONESHOT_DISABLE:
  325. oneshot_disable();
  326. break;
  327. #endif
  328. }
  329. }
  330. return process_action_kb(record);
  331. }
  332. void set_single_persistent_default_layer(uint8_t default_layer) {
  333. #if defined(AUDIO_ENABLE) && defined(DEFAULT_LAYER_SONGS)
  334. PLAY_SONG(default_layer_songs[default_layer]);
  335. #endif
  336. eeconfig_update_default_layer((layer_state_t)1 << default_layer);
  337. default_layer_set((layer_state_t)1 << default_layer);
  338. }
  339. layer_state_t update_tri_layer_state(layer_state_t state, uint8_t layer1, uint8_t layer2, uint8_t layer3) {
  340. layer_state_t mask12 = ((layer_state_t)1 << layer1) | ((layer_state_t)1 << layer2);
  341. layer_state_t mask3 = (layer_state_t)1 << layer3;
  342. return (state & mask12) == mask12 ? (state | mask3) : (state & ~mask3);
  343. }
  344. void update_tri_layer(uint8_t layer1, uint8_t layer2, uint8_t layer3) {
  345. layer_state_set(update_tri_layer_state(layer_state, layer1, layer2, layer3));
  346. }
  347. // TODO: remove legacy api
  348. void matrix_init_quantum() {
  349. matrix_init_kb();
  350. }
  351. void matrix_scan_quantum() {
  352. matrix_scan_kb();
  353. }
  354. //------------------------------------------------------------------------------
  355. // Override these functions in your keymap file to play different tunes on
  356. // different events such as startup and bootloader jump
  357. __attribute__((weak)) void startup_user() {}
  358. __attribute__((weak)) void shutdown_user() {}
  359. void suspend_power_down_quantum(void) {
  360. suspend_power_down_kb();
  361. #ifndef NO_SUSPEND_POWER_DOWN
  362. // Turn off backlight
  363. # ifdef BACKLIGHT_ENABLE
  364. backlight_set(0);
  365. # endif
  366. # ifdef LED_MATRIX_ENABLE
  367. led_matrix_task();
  368. # endif
  369. # ifdef RGB_MATRIX_ENABLE
  370. rgb_matrix_task();
  371. # endif
  372. // Turn off LED indicators
  373. led_suspend();
  374. // Turn off audio
  375. # ifdef AUDIO_ENABLE
  376. stop_all_notes();
  377. # endif
  378. // Turn off underglow
  379. # if defined(RGBLIGHT_SLEEP) && defined(RGBLIGHT_ENABLE)
  380. rgblight_suspend();
  381. # endif
  382. # if defined(LED_MATRIX_ENABLE)
  383. led_matrix_set_suspend_state(true);
  384. # endif
  385. # if defined(RGB_MATRIX_ENABLE)
  386. rgb_matrix_set_suspend_state(true);
  387. # endif
  388. # ifdef OLED_ENABLE
  389. oled_off();
  390. # endif
  391. # ifdef ST7565_ENABLE
  392. st7565_off();
  393. # endif
  394. # if defined(POINTING_DEVICE_ENABLE)
  395. // run to ensure scanning occurs while suspended
  396. pointing_device_task();
  397. # endif
  398. #endif
  399. }
  400. __attribute__((weak)) void suspend_wakeup_init_quantum(void) {
  401. // Turn on backlight
  402. #ifdef BACKLIGHT_ENABLE
  403. backlight_init();
  404. #endif
  405. // Restore LED indicators
  406. led_wakeup();
  407. // Wake up underglow
  408. #if defined(RGBLIGHT_SLEEP) && defined(RGBLIGHT_ENABLE)
  409. rgblight_wakeup();
  410. #endif
  411. #if defined(LED_MATRIX_ENABLE)
  412. led_matrix_set_suspend_state(false);
  413. #endif
  414. #if defined(RGB_MATRIX_ENABLE)
  415. rgb_matrix_set_suspend_state(false);
  416. #endif
  417. suspend_wakeup_init_kb();
  418. }
  419. /** \brief converts unsigned integers into char arrays
  420. *
  421. * Takes an unsigned integer and converts that value into an equivalent char array
  422. * A padding character may be specified, ' ' for leading spaces, '0' for leading zeros.
  423. */
  424. const char *get_numeric_str(char *buf, size_t buf_len, uint32_t curr_num, char curr_pad) {
  425. buf[buf_len - 1] = '\0';
  426. for (size_t i = 0; i < buf_len - 1; ++i) {
  427. char c = '0' + curr_num % 10;
  428. buf[buf_len - 2 - i] = (c == '0' && i == 0) ? '0' : (curr_num > 0 ? c : curr_pad);
  429. curr_num /= 10;
  430. }
  431. return buf;
  432. }
  433. /** \brief converts uint8_t into char array
  434. *
  435. * Takes an uint8_t, and uses an internal static buffer to render that value into a char array
  436. * A padding character may be specified, ' ' for leading spaces, '0' for leading zeros.
  437. *
  438. * NOTE: Subsequent invocations will reuse the same static buffer and overwrite the previous
  439. * contents. Use the result immediately, instead of caching it.
  440. */
  441. const char *get_u8_str(uint8_t curr_num, char curr_pad) {
  442. static char buf[4] = {0};
  443. static uint8_t last_num = 0xFF;
  444. static char last_pad = '\0';
  445. if (last_num == curr_num && last_pad == curr_pad) {
  446. return buf;
  447. }
  448. last_num = curr_num;
  449. last_pad = curr_pad;
  450. return get_numeric_str(buf, sizeof(buf), curr_num, curr_pad);
  451. }
  452. /** \brief converts uint16_t into char array
  453. *
  454. * Takes an uint16_t, and uses an internal static buffer to render that value into a char array
  455. * A padding character may be specified, ' ' for leading spaces, '0' for leading zeros.
  456. *
  457. * NOTE: Subsequent invocations will reuse the same static buffer and overwrite the previous
  458. * contents. Use the result immediately, instead of caching it.
  459. */
  460. const char *get_u16_str(uint16_t curr_num, char curr_pad) {
  461. static char buf[6] = {0};
  462. static uint16_t last_num = 0xFF;
  463. static char last_pad = '\0';
  464. if (last_num == curr_num && last_pad == curr_pad) {
  465. return buf;
  466. }
  467. last_num = curr_num;
  468. last_pad = curr_pad;
  469. return get_numeric_str(buf, sizeof(buf), curr_num, curr_pad);
  470. }