Add kb and user level keyboard initialization functions (#3113)
* Add suspend functions * Disable RGB code if it's disabled * Add keyboard_init functions * Change where references so it will compile * Wrong command chained in wake up kb function * Fix non-feature file changes * Add documentation * Re-add matrix init docs * add rgblight code to example * Remove suspend code * Clean up docs * Fix docs * Fix suspend code * more doc fixes * change function to startup_* rather than keyboard_init_ * fix spelling error * fix up docs to finish removing keyboard_init * Use Pre and Post init functions * Update Documenation * Remove changes to my keymap and userspace code * Cleanup * Revert changes to extra files * Forgot a semicolon * Make sure all protocols call keyboard_setup * Cleanup functions * Unset startup_user * Remove changes from division keyboard * Readd startup_user function * Remove all to startup_user * Update docs/custom_quantum_functions.md Co-Authored-By: drashna <drashna@live.com> * Update docs/custom_quantum_functions.md Co-Authored-By: drashna <drashna@live.com> * Add suggestion line * Rebase fixes * Update documentation to be more useful/accurate * Cleanup of documentation * Fix spacing inconsistency * Revert unexpected change to keymap
This commit is contained in:
parent
40e67a3074
commit
cc5c6b449a
|
@ -165,18 +165,35 @@ In addition, it is possible to specify the brightness level of all LEDs with `er
|
||||||
|
|
||||||
Ergodox boards also define `LED_BRIGHTNESS_LO` for the lowest brightness and `LED_BRIGHTNESS_HI` for the highest brightness (which is the default).
|
Ergodox boards also define `LED_BRIGHTNESS_LO` for the lowest brightness and `LED_BRIGHTNESS_HI` for the highest brightness (which is the default).
|
||||||
|
|
||||||
# Matrix Initialization Code
|
# Keyboard Initialization Code
|
||||||
|
|
||||||
Before a keyboard can be used the hardware must be initialized. QMK handles initialization of the keyboard matrix itself, but if you have other hardware like LEDs or i²c controllers you will need to set up that hardware before it can be used.
|
There are several steps in the keyboard initialization process. Depending on what you want to do, it will influence which function you should use.
|
||||||
|
|
||||||
|
These are the three main initialization functions, listed in the order that they're called.
|
||||||
|
|
||||||
### Example `matrix_init_user()` Implementation
|
* `keyboard_pre_init_*` - Happens before most anything is started. Good for hardware setup that you want running very early.
|
||||||
|
* `matrix_init_*` - Happens midway through the firmware's startup process. Hardware is initialized, but features may not be yet.
|
||||||
|
* `keyboard_post_init_*` - Happens at the end of the firmware's startup process. This is where you'd want to put "customization" code, for the most part.
|
||||||
|
|
||||||
|
!> For most people, the `keyboard_post_init_user` function is what you want to call. For instance, this is where you want to set up things for RGB Underglow.
|
||||||
|
|
||||||
|
## Keyboard Pre Initialization code
|
||||||
|
|
||||||
|
This runs very early during startup, even before the USB has been started.
|
||||||
|
|
||||||
|
Shortly after this, the matrix is initialized.
|
||||||
|
|
||||||
|
For most users, this shouldn't be used, as it's primarily for hardware oriented initialization.
|
||||||
|
|
||||||
|
However, if you have hardware stuff that you need initialized, this is the best place for it (such as initializing LED pins).
|
||||||
|
|
||||||
|
### Example `keyboard_pre_init_user()` Implementation
|
||||||
|
|
||||||
This example, at the keyboard level, sets up B1, B2, and B3 as LED pins.
|
This example, at the keyboard level, sets up B1, B2, and B3 as LED pins.
|
||||||
|
|
||||||
```c
|
```c
|
||||||
void matrix_init_user(void) {
|
void keyboard_pre_init_user(void) {
|
||||||
// Call the keymap level matrix init.
|
// Call the keyboard pre init code.
|
||||||
|
|
||||||
// Set our LED pins as output
|
// Set our LED pins as output
|
||||||
DDRB |= (1<<1);
|
DDRB |= (1<<1);
|
||||||
|
@ -185,11 +202,47 @@ void matrix_init_user(void) {
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### `keyboard_pre_init_*` Function Documentation
|
||||||
|
|
||||||
|
* Keyboard/Revision: `void keyboard_pre_init_kb(void)`
|
||||||
|
* Keymap: `void keyboard_pre_init_user(void)`
|
||||||
|
|
||||||
|
## Matrix Initialization Code
|
||||||
|
|
||||||
|
This is called when the matrix is initialized, and after some of the hardware has been set up, but before many of the features have been initialized.
|
||||||
|
|
||||||
|
This is useful for setting up stuff that you may need elsewhere, but isn't hardware related nor is dependant on where it's started.
|
||||||
|
|
||||||
|
|
||||||
### `matrix_init_*` Function Documentation
|
### `matrix_init_*` Function Documentation
|
||||||
|
|
||||||
* Keyboard/Revision: `void matrix_init_kb(void)`
|
* Keyboard/Revision: `void matrix_init_kb(void)`
|
||||||
* Keymap: `void matrix_init_user(void)`
|
* Keymap: `void matrix_init_user(void)`
|
||||||
|
|
||||||
|
|
||||||
|
## Keyboard Post Initialization code
|
||||||
|
|
||||||
|
This is ran as the very last task in the keyboard initialization process. This is useful if you want to make changes to certain features, as they should be initialized by this point.
|
||||||
|
|
||||||
|
|
||||||
|
### Example `keyboard_post_init_user()` Implementation
|
||||||
|
|
||||||
|
This example, running after everything else has initialized, sets up the rgb underglow configuration.
|
||||||
|
|
||||||
|
```c
|
||||||
|
void keyboard_post_init_user(void) {
|
||||||
|
// Call the post init code.
|
||||||
|
rgblight_enable_noeeprom(); // enables Rgb, without saving settings
|
||||||
|
rgblight_sethsv_noeeprom(180, 255, 255): // sets the color to teal/cyan without saving
|
||||||
|
rgblight_mode_noeeprom(RGBLIGHT_MODE_BREATHING + 3); // sets mode to Fast breathing without saving
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### `keyboard_post_init_*` Function Documentation
|
||||||
|
|
||||||
|
* Keyboard/Revision: `void keyboard_post_init_kb(void)`
|
||||||
|
* Keymap: `void keyboard_post_init_user(void)`
|
||||||
|
|
||||||
# Matrix Scanning Code
|
# Matrix Scanning Code
|
||||||
|
|
||||||
Whenever possible you should customize your keyboard by using `process_record_*()` and hooking into events that way, to ensure that your code does not have a negative performance impact on your keyboard. However, in rare cases it is necessary to hook into the matrix scanning. Be extremely careful with the performance of code in these functions, as it will be called at least 10 times per second.
|
Whenever possible you should customize your keyboard by using `process_record_*()` and hooking into events that way, to ensure that your code does not have a negative performance impact on your keyboard. However, in rare cases it is necessary to hook into the matrix scanning. Be extremely careful with the performance of code in these functions, as it will be called at least 10 times per second.
|
||||||
|
@ -229,10 +282,9 @@ void suspend_wakeup_init_user(void)
|
||||||
{
|
{
|
||||||
rgb_matrix_set_suspend_state(false);
|
rgb_matrix_set_suspend_state(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### `keyboard_init_*` Function Documentation
|
### Keyboard suspend/wake Function Documentation
|
||||||
|
|
||||||
* Keyboard/Revision: `void suspend_power_down_kb(void)` and `void suspend_wakeup_init_user(void)`
|
* Keyboard/Revision: `void suspend_power_down_kb(void)` and `void suspend_wakeup_init_user(void)`
|
||||||
* Keymap: `void suspend_power_down_kb(void)` and `void suspend_wakeup_init_user(void)`
|
* Keymap: `void suspend_power_down_kb(void)` and `void suspend_wakeup_init_user(void)`
|
||||||
|
@ -285,7 +337,7 @@ Keep in mind that EEPROM has a limited number of writes. While this is very high
|
||||||
|
|
||||||
* If you don't understand the example, then you may want to avoid using this feature, as it is rather complicated.
|
* If you don't understand the example, then you may want to avoid using this feature, as it is rather complicated.
|
||||||
|
|
||||||
### Example Implementation
|
### Example Implementation
|
||||||
|
|
||||||
This is an example of how to add settings, and read and write it. We're using the user keymap for the example here. This is a complex function, and has a lot going on. In fact, it uses a lot of the above functions to work!
|
This is an example of how to add settings, and read and write it. We're using the user keymap for the example here. This is a complex function, and has a lot going on. In fact, it uses a lot of the above functions to work!
|
||||||
|
|
||||||
|
|
|
@ -142,20 +142,20 @@ static void power_down(uint8_t wdto) {
|
||||||
#endif
|
#endif
|
||||||
suspend_power_down_kb();
|
suspend_power_down_kb();
|
||||||
|
|
||||||
// TODO: more power saving
|
// TODO: more power saving
|
||||||
// See PicoPower application note
|
// See PicoPower application note
|
||||||
// - I/O port input with pullup
|
// - I/O port input with pullup
|
||||||
// - prescale clock
|
// - prescale clock
|
||||||
// - BOD disable
|
// - BOD disable
|
||||||
// - Power Reduction Register PRR
|
// - Power Reduction Register PRR
|
||||||
set_sleep_mode(SLEEP_MODE_PWR_DOWN);
|
set_sleep_mode(SLEEP_MODE_PWR_DOWN);
|
||||||
sleep_enable();
|
sleep_enable();
|
||||||
sei();
|
sei();
|
||||||
sleep_cpu();
|
sleep_cpu();
|
||||||
sleep_disable();
|
sleep_disable();
|
||||||
|
|
||||||
// Disable watchdog after sleep
|
// Disable watchdog after sleep
|
||||||
wdt_disable();
|
wdt_disable();
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -139,6 +139,40 @@ __attribute__ ((weak))
|
||||||
void matrix_setup(void) {
|
void matrix_setup(void) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** \brief keyboard_pre_init_user
|
||||||
|
*
|
||||||
|
* FIXME: needs doc
|
||||||
|
*/
|
||||||
|
__attribute__ ((weak))
|
||||||
|
void keyboard_pre_init_user(void) { }
|
||||||
|
|
||||||
|
/** \brief keyboard_pre_init_kb
|
||||||
|
*
|
||||||
|
* FIXME: needs doc
|
||||||
|
*/
|
||||||
|
__attribute__ ((weak))
|
||||||
|
void keyboard_pre_init_kb(void) {
|
||||||
|
keyboard_pre_init_user();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** \brief keyboard_post_init_user
|
||||||
|
*
|
||||||
|
* FIXME: needs doc
|
||||||
|
*/
|
||||||
|
|
||||||
|
__attribute__ ((weak))
|
||||||
|
void keyboard_post_init_user() {}
|
||||||
|
|
||||||
|
/** \brief keyboard_post_init_kb
|
||||||
|
*
|
||||||
|
* FIXME: needs doc
|
||||||
|
*/
|
||||||
|
|
||||||
|
__attribute__ ((weak))
|
||||||
|
void keyboard_post_init_kb(void) {
|
||||||
|
keyboard_post_init_user();
|
||||||
|
}
|
||||||
|
|
||||||
/** \brief keyboard_setup
|
/** \brief keyboard_setup
|
||||||
*
|
*
|
||||||
* FIXME: needs doc
|
* FIXME: needs doc
|
||||||
|
@ -146,6 +180,7 @@ void matrix_setup(void) {
|
||||||
void keyboard_setup(void) {
|
void keyboard_setup(void) {
|
||||||
disable_jtag();
|
disable_jtag();
|
||||||
matrix_setup();
|
matrix_setup();
|
||||||
|
keyboard_pre_init_kb();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** \brief is_keyboard_master
|
/** \brief is_keyboard_master
|
||||||
|
@ -199,6 +234,7 @@ void keyboard_init(void) {
|
||||||
#if defined(NKRO_ENABLE) && defined(FORCE_NKRO)
|
#if defined(NKRO_ENABLE) && defined(FORCE_NKRO)
|
||||||
keymap_config.nkro = 1;
|
keymap_config.nkro = 1;
|
||||||
#endif
|
#endif
|
||||||
|
keyboard_post_init_kb(); /* Always keep this last */
|
||||||
}
|
}
|
||||||
|
|
||||||
/** \brief Keyboard task: Do keyboard routine jobs
|
/** \brief Keyboard task: Do keyboard routine jobs
|
||||||
|
|
|
@ -70,6 +70,11 @@ void keyboard_set_leds(uint8_t leds);
|
||||||
/* it runs whenever code has to behave differently on a slave */
|
/* it runs whenever code has to behave differently on a slave */
|
||||||
bool is_keyboard_master(void);
|
bool is_keyboard_master(void);
|
||||||
|
|
||||||
|
void keyboard_pre_init_kb(void);
|
||||||
|
void keyboard_pre_init_user(void);
|
||||||
|
void keyboard_post_init_kb(void);
|
||||||
|
void keyboard_post_init_user(void);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -42,13 +42,13 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
int main(void)
|
int main(void)
|
||||||
{
|
{
|
||||||
|
|
||||||
CPU_PRESCALE(0);
|
CPU_PRESCALE(0);
|
||||||
|
|
||||||
// DDRD = _BV(PD5);
|
// DDRD = _BV(PD5);
|
||||||
// DDRB = _BV(PB0);
|
// DDRB = _BV(PB0);
|
||||||
|
|
||||||
// PORTD = _BV(PD5);
|
// PORTD = _BV(PD5);
|
||||||
// PORTB = _BV(PB0);
|
// PORTB = _BV(PB0);
|
||||||
|
|
||||||
|
@ -59,22 +59,23 @@ int main(void)
|
||||||
// while (!usb_configured()) /* wait */
|
// while (!usb_configured()) /* wait */
|
||||||
|
|
||||||
|
|
||||||
|
keyboard_setup();
|
||||||
|
|
||||||
dprintf("Initializing keyboard...\n");
|
dprintf("Initializing keyboard...\n");
|
||||||
keyboard_init();
|
keyboard_init();
|
||||||
|
|
||||||
// This implementation is pretty simplistic... if the USB connection
|
// This implementation is pretty simplistic... if the USB connection
|
||||||
// is not configured, choose the Bluefruit, otherwise use USB
|
// is not configured, choose the Bluefruit, otherwise use USB
|
||||||
// Definitely would prefer to have this driven by an input pin and make
|
// Definitely would prefer to have this driven by an input pin and make
|
||||||
// it switch dynamically - BCG
|
// it switch dynamically - BCG
|
||||||
// if (!usb_configured()) {
|
// if (!usb_configured()) {
|
||||||
|
|
||||||
// // Send power to Bluefruit... Adafruit says it takes 27 mA, I think
|
// // Send power to Bluefruit... Adafruit says it takes 27 mA, I think
|
||||||
// // the pins should provide 40 mA, but just in case I switch the
|
// // the pins should provide 40 mA, but just in case I switch the
|
||||||
// // Bluefruit using a transistor - BCG
|
// // Bluefruit using a transistor - BCG
|
||||||
// DDRB = _BV(PB6);
|
// DDRB = _BV(PB6);
|
||||||
// PORTB |= _BV(PB6);
|
// PORTB |= _BV(PB6);
|
||||||
|
|
||||||
dprintf("Setting host driver to bluefruit...\n");
|
dprintf("Setting host driver to bluefruit...\n");
|
||||||
host_set_driver(bluefruit_driver());
|
host_set_driver(bluefruit_driver());
|
||||||
|
|
||||||
|
@ -131,7 +132,7 @@ int main(void)
|
||||||
// usb_remote_wakeup();
|
// usb_remote_wakeup();
|
||||||
// }
|
// }
|
||||||
// }
|
// }
|
||||||
// keyboard_task();
|
// keyboard_task();
|
||||||
// }
|
// }
|
||||||
// }
|
// }
|
||||||
|
|
||||||
|
|
|
@ -119,6 +119,8 @@ int main(void) {
|
||||||
// TESTING
|
// TESTING
|
||||||
// chThdCreateStatic(waThread1, sizeof(waThread1), NORMALPRIO, Thread1, NULL);
|
// chThdCreateStatic(waThread1, sizeof(waThread1), NORMALPRIO, Thread1, NULL);
|
||||||
|
|
||||||
|
keyboard_setup();
|
||||||
|
|
||||||
/* Init USB */
|
/* Init USB */
|
||||||
init_usb_driver(&USB_DRIVER);
|
init_usb_driver(&USB_DRIVER);
|
||||||
|
|
||||||
|
|
|
@ -56,6 +56,7 @@ int main(void)
|
||||||
#ifndef NO_UART
|
#ifndef NO_UART
|
||||||
uart_init(UART_BAUD_RATE);
|
uart_init(UART_BAUD_RATE);
|
||||||
#endif
|
#endif
|
||||||
|
keyboard_setup();
|
||||||
|
|
||||||
keyboard_init();
|
keyboard_init();
|
||||||
host_set_driver(vusb_driver());
|
host_set_driver(vusb_driver());
|
||||||
|
|
Loading…
Reference in New Issue