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.

217 lines
9.0 KiB

  1. # Squeezing the most out of AVR
  2. AVR is severely resource-constrained, and as QMK continues to grow, it is approaching a point where support for AVR may need to be moved to legacy status as newer development is unable to fit into those constraints.
  3. However, if you need to reduce the compiled size of your firmware, there are a number of options to do so.
  4. ## `rules.mk` Settings
  5. First and foremost is enabling link time optimization. To do so, add this to your rules.mk:
  6. ```make
  7. LTO_ENABLE = yes
  8. ```
  9. This will cause the final step to take longer, but should get you a smaller compiled size. This also disables Action Functions, and Action Macros, both of which are deprecated.
  10. This will get you the most savings, in most situations.
  11. From there, disabling extraneous systems will help -- e.g.:
  12. ```make
  13. CONSOLE_ENABLE = no
  14. COMMAND_ENABLE = no
  15. MOUSEKEY_ENABLE = no
  16. EXTRAKEY_ENABLE = no
  17. ```
  18. This disables some of the functionality that you may not need. But note that extrakeys disables stuff like the media keys and system volume control.
  19. If that isn't enough to get your firmware down to size, then there are some additional features that you can disable:
  20. ```make
  21. SPACE_CADET_ENABLE = no
  22. GRAVE_ESC_ENABLE = no
  23. MAGIC_ENABLE = no
  24. ```
  25. These features are enabled by default, but they may not be needed. Double check to make sure. The [Magic Keycodes](keycodes_magic.md) are the largest and control things like NKRO toggling, GUI and ALT/CTRL swapping, etc. Disabling them will disable those functions. See [Magic Functions](#magic-functions) for disabling related functions.
  26. If you use `sprintf` or `snprintf` functions you can save around ~400 Bytes by enabling this option.
  27. ```make
  28. AVR_USE_MINIMAL_PRINTF = yes
  29. ```
  30. This will include smaller implementations from AVRs libc into your Firmware. They are [not fully featured](https://www.nongnu.org/avr-libc/user-manual/group__avr__stdio.html#gaa3b98c0d17b35642c0f3e4649092b9f1), for instance zero padding and field width specifiers are not supported. So if you use `sprintf` or `snprintf` like this:
  31. ```c
  32. sprintf(wpm_str, "%03d", get_current_wpm());
  33. snprintf(keylog_str, sizeof(keylog_str), "%dx%d, k%2d : %c");
  34. ```
  35. you will still need the standard implementation.
  36. ## `config.h` Settings
  37. If you've done all of that, and you don't want to disable features like RGB, Audio, OLEDs, etc, there are some additional options that you can add to your config.h that can help.
  38. Starting with Lock Key support. If you have a Cherry MX Lock switch (lucky you!), you don't want to do this. But chances are, you don't. In that case, add this to your `config.h`:
  39. ```c
  40. #undef LOCKING_SUPPORT_ENABLE
  41. #undef LOCKING_RESYNC_ENABLE
  42. ```
  43. Oneshots. If you're not using these, you can disable the feature by adding this to your `config.h`:
  44. ```c
  45. #define NO_ACTION_ONESHOT
  46. ```
  47. The same with tapping keys (mod tap, layer tap, etc)
  48. ```c
  49. #define NO_ACTION_TAPPING
  50. ```
  51. ## Audio Settings
  52. If you're using the Audio feature, by default that includes the music mode feature. This tranlates matrix positions into notes. It's neat for sure, but most likely, you're not using it. You can disable it by adding this to your `config.h`:
  53. ```c
  54. #define NO_MUSIC_MODE
  55. ```
  56. And by adding this to your `rules.mk`
  57. ```make
  58. MUSIC_ENABLE = no
  59. ```
  60. ## Layers
  61. There are also some options for layers, that can reduce the firmware size. All of these settings are for your `config.h`.
  62. You can limit the number of layers that the firmware uses -- if you're using up to 8 layers in total:
  63. ```c
  64. #define LAYER_STATE_8BIT
  65. ```
  66. or if you require up to 16 layers instead:
  67. ```c
  68. #define LAYER_STATE_16BIT
  69. ```
  70. Or if you're not using layers at all, you can outright remove the functionality altogether:
  71. ```c
  72. #define NO_ACTION_LAYER
  73. ```
  74. ## Magic Functions
  75. There are two `__attribute__ ((weak))` placeholder functions available to customize magic keycodes. If you are not using that feature to swap keycodes, such as backslash with backspace, add the following to your `keymap.c` or user space code:
  76. ```c
  77. uint16_t keycode_config(uint16_t keycode) {
  78. return keycode;
  79. }
  80. ```
  81. Likewise, if you are not using magic keycodes to swap modifiers, such as Control with GUI, add the following to your `keymap.c` or user space code:
  82. ```c
  83. uint8_t mod_config(uint8_t mod) {
  84. return mod;
  85. }
  86. ```
  87. Both of them will overwrite the placeholder functions with a simple return statement to reduce firmware size.
  88. ## OLED tweaks
  89. One place you can save a bunch of space here is by not using `sprintf` or `snprintf`. This function call takes up ~1.5kB of firmware space, and can be rewritten. For instance, WPM uses this a lot.
  90. You can convert this:
  91. ```c
  92. // OLD CODE
  93. char wpm_str[4] = {0};
  94. sprintf(wpm_str, "WPM: %03d", get_current_wpm());
  95. oled_write(wpm_str, ' '), false);
  96. ```
  97. into this:
  98. ```c
  99. // NEW CODE
  100. oled_write_P(PSTR("WPM: "), false);
  101. oled_write(get_u8_str(get_current_wpm(), ' '), false);
  102. ```
  103. which outputs `WPM: 5`. Or this:
  104. ```c
  105. // NEW CODE
  106. oled_write_P(PSTR("WPM: "), false);
  107. oled_write(get_u8_str(get_current_wpm(), '0'), false);
  108. ```
  109. which outputs `WPM: 005`.
  110. ## RGB Settings
  111. If you're using RGB on your board, both RGB Light (Underglow) and RGB Matrix (per key RGB) now require defines to enable different animations -- some keyboards enable a lot of animations by default, so you can generally gain back some space by disabling specific animations if you don't use them. For RGB Light you can disable these in your keymap's `config.h`:
  112. ```c
  113. #undef RGBLIGHT_ANIMATIONS
  114. #undef RGBLIGHT_EFFECT_BREATHING
  115. #undef RGBLIGHT_EFFECT_RAINBOW_MOOD
  116. #undef RGBLIGHT_EFFECT_RAINBOW_SWIRL
  117. #undef RGBLIGHT_EFFECT_SNAKE
  118. #undef RGBLIGHT_EFFECT_KNIGHT
  119. #undef RGBLIGHT_EFFECT_CHRISTMAS
  120. #undef RGBLIGHT_EFFECT_STATIC_GRADIENT
  121. #undef RGBLIGHT_EFFECT_RGB_TEST
  122. #undef RGBLIGHT_EFFECT_ALTERNATING
  123. #undef RGBLIGHT_EFFECT_TWINKLE
  124. ```
  125. For RGB Matrix, these need to be explicitly enabled as well. To disable any that were enabled by the keyboard, add one or more of these to your keymap's `config.h`:
  126. ```c
  127. #undef ENABLE_RGB_MATRIX_ALPHAS_MODS
  128. #undef ENABLE_RGB_MATRIX_GRADIENT_UP_DOWN
  129. #undef ENABLE_RGB_MATRIX_GRADIENT_LEFT_RIGHT
  130. #undef ENABLE_RGB_MATRIX_BREATHING
  131. #undef ENABLE_RGB_MATRIX_BAND_SAT
  132. #undef ENABLE_RGB_MATRIX_BAND_VAL
  133. #undef ENABLE_RGB_MATRIX_BAND_PINWHEEL_SAT
  134. #undef ENABLE_RGB_MATRIX_BAND_PINWHEEL_VAL
  135. #undef ENABLE_RGB_MATRIX_BAND_SPIRAL_SAT
  136. #undef ENABLE_RGB_MATRIX_BAND_SPIRAL_VAL
  137. #undef ENABLE_RGB_MATRIX_CYCLE_ALL
  138. #undef ENABLE_RGB_MATRIX_CYCLE_LEFT_RIGHT
  139. #undef ENABLE_RGB_MATRIX_CYCLE_UP_DOWN
  140. #undef ENABLE_RGB_MATRIX_RAINBOW_MOVING_CHEVRON
  141. #undef ENABLE_RGB_MATRIX_CYCLE_OUT_IN
  142. #undef ENABLE_RGB_MATRIX_CYCLE_OUT_IN_DUAL
  143. #undef ENABLE_RGB_MATRIX_CYCLE_PINWHEEL
  144. #undef ENABLE_RGB_MATRIX_CYCLE_SPIRAL
  145. #undef ENABLE_RGB_MATRIX_DUAL_BEACON
  146. #undef ENABLE_RGB_MATRIX_RAINBOW_BEACON
  147. #undef ENABLE_RGB_MATRIX_RAINBOW_PINWHEELS
  148. #undef ENABLE_RGB_MATRIX_FLOWER_BLOOMING
  149. #undef ENABLE_RGB_MATRIX_RAINDROPS
  150. #undef ENABLE_RGB_MATRIX_JELLYBEAN_RAINDROPS
  151. #undef ENABLE_RGB_MATRIX_HUE_BREATHING
  152. #undef ENABLE_RGB_MATRIX_HUE_PENDULUM
  153. #undef ENABLE_RGB_MATRIX_HUE_WAVE
  154. #undef ENABLE_RGB_MATRIX_PIXEL_FRACTAL
  155. #undef ENABLE_RGB_MATRIX_PIXEL_FLOW
  156. #undef ENABLE_RGB_MATRIX_PIXEL_RAIN
  157. #undef ENABLE_RGB_MATRIX_TYPING_HEATMAP
  158. #undef ENABLE_RGB_MATRIX_DIGITAL_RAIN
  159. #undef ENABLE_RGB_MATRIX_SOLID_REACTIVE_SIMPLE
  160. #undef ENABLE_RGB_MATRIX_SOLID_REACTIVE
  161. #undef ENABLE_RGB_MATRIX_SOLID_REACTIVE_WIDE
  162. #undef ENABLE_RGB_MATRIX_SOLID_REACTIVE_MULTIWIDE
  163. #undef ENABLE_RGB_MATRIX_SOLID_REACTIVE_CROSS
  164. #undef ENABLE_RGB_MATRIX_SOLID_REACTIVE_MULTICROSS
  165. #undef ENABLE_RGB_MATRIX_SOLID_REACTIVE_NEXUS
  166. #undef ENABLE_RGB_MATRIX_SOLID_REACTIVE_MULTINEXUS
  167. #undef ENABLE_RGB_MATRIX_SPLASH
  168. #undef ENABLE_RGB_MATRIX_MULTISPLASH
  169. #undef ENABLE_RGB_MATRIX_SOLID_SPLASH
  170. #undef ENABLE_RGB_MATRIX_SOLID_MULTISPLASH
  171. ```
  172. # Final Thoughts
  173. If you've done all of this, and your firmware is still too large, then it's time. It's time to consider making the switch to ARM. Unfortunately, right now is the worst possible time for that, due to the silicon shortage, and supply chain issues. Getting an ARM chip is difficult, at best, and significantly overpriced, at worst.
  174. -- Drashna
  175. That said, there are a number of Pro Micro replacements with ARM controllers:
  176. * [Proton C](https://qmk.fm/proton-c/) (out of stock)
  177. * [Bonsai C](https://github.com/customMK/Bonsai-C) (Open Source, DIY/PCBA)
  178. * [STeMCell](https://github.com/megamind4089/STeMCell) (Open Source, DIY/PCBA)
  179. * [Adafruit KB2040](https://learn.adafruit.com/adafruit-kb2040)
  180. * [SparkFun Pro Micro - RP2040](https://www.sparkfun.com/products/18288)
  181. * [Blok](https://boardsource.xyz/store/628b95b494dfa308a6581622)
  182. * [Elite-Pi](https://keeb.io/products/elite-pi-usb-c-pro-micro-replacement-rp2040)
  183. * [0xCB Helios](https://keeb.supply/products/0xcb-helios) ([Open Source](https://github.com/0xCB-dev/0xCB-Helios), DIY/PCBA/Shop)
  184. * [Liatris](https://splitkb.com/products/liatris)
  185. * [Imera](https://splitkb.com/products/imera)
  186. * [Michi](https://github.com/ci-bus/michi-promicro-rp2040)
  187. There are other, non-Pro Micro compatible boards out there. The most popular being:
  188. * [WeAct Blackpill F411](https://www.aliexpress.com/item/1005001456186625.html) (~$6 USD)