* keycode_string(): Format keycodes as strings. This adds the `keycode_string()` function described in https://getreuer.info/posts/keyboards/keycode-string/index.html as a core feature. * Fix formatting. * keycode_string review revisions. * Rename keycode_string() -> get_keycode_string() for consistency with existing string utils like get_u8_str(). * Revise custom keycode names with separate _user and _kb tables. * Correct indent in builddefs/generic_features.mk. Co-authored-by: Ryan <fauxpark@gmail.com> * Add KC_NUHS, KC_NUBS, and KC_CAPS. * Fix linking error with custom names. * Attempt at simplifying interface. * Formatting fix. * Several fixes and revisions. * Don't use PSTR in KEYCODE_STRING_NAME, since this fails to build on AVR. Store custom names in RAM. * Revise the internal table of common keycode names to use its own storage representation, still in PROGMEM, and now more efficiently stored flat in 8 bytes per entry. * Support Swap Hands keycodes and a few other keycodes. * Revert "Formatting fix." This reverts commit 2a2771068c7ee545ffac4103aa07e847a9ec3816. * Revert "Attempt at simplifying interface." This reverts commit 8eaf67de76e75bc92d106a8b0decc893fbc65fa5. * Simplify custom names API by sigprof's suggestion. * Support more keycodes. * Add QK_LOCK keycode. * Add Secure keycodes. * Add Joystick keycodes. * Add Programmable Button keycodes. * Add macro MC_ keycodes. * For remaining keys in known code ranges, stringify them as "QK_<feature>+<number>". For instance, "QK_MIDI+7". * Bug fix and a few improvements. * Fix missing right-hand bit when displaying 5-bit mods numerically. * Support KC_HYPR, KC_MEH, HYPR_T(kc), MEH_T(kc). * Exclude one-shot keycodes when NO_ACTION_ONESHOT is defined. --------- Co-authored-by: Ryan <fauxpark@gmail.com>
		
			
				
	
	
		
			107 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
			
		
		
	
	
			107 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
| # Unit Testing
 | |
| 
 | |
| If you are new to unit testing, then you can find many good resources on internet. However most of it is scattered around in small pieces here and there, and there's also many different opinions, so I won't give any recommendations.
 | |
| 
 | |
| Instead I recommend these two books, explaining two different styles of Unit Testing in detail.
 | |
| 
 | |
| * "Test Driven Development: By Example: Kent Beck"
 | |
| * "Growing Object-Oriented Software, Guided By Tests: Steve Freeman, Nat Pryce"
 | |
| 
 | |
| If you prefer videos there are Uncle Bob's [Clean Coders Videos](https://cleancoders.com/), which unfortunately cost quite a bit, especially if you want to watch many of them. But James Shore has a free [Let's Play](https://www.jamesshore.com/Blog/Lets-Play) video series.
 | |
| 
 | |
| ## Google Test and Google Mock
 | |
| It's possible to Unit Test your code using [Google Test](https://github.com/google/googletest). The Google Test framework also includes another component for writing testing mocks and stubs, called "Google Mock". For information how to write the actual tests, please refer to the documentation on that site.
 | |
| 
 | |
| ## Use of C++
 | |
| 
 | |
| Note that Google Test and therefore any test has to be written in C++, even if the rest of the QMK codebases is written in C. This should hopefully not be a problem even if you don't know any C++, since there's quite clear documentation and examples of the required C++ features, and you can write the rest of the test code almost as you would write normal C. Note that some compiler errors which you might get can look quite scary, but just read carefully what it says, and you should be ok.
 | |
| 
 | |
| One thing to remember, is that you have to append `extern "C"` around all of your C file includes.
 | |
| 
 | |
| ## Adding Tests for New or Existing Features
 | |
| 
 | |
| If you want to unit test a feature, take a look at some of the existing tests, for example those in the `quantum/sequencer/tests` folder. Then follow the steps below to create a similar structure.
 | |
| 
 | |
| 1. If it doesn't already exist, add a test subfolder to the folder containing the feature.
 | |
| 2. Create a `testlist.mk` and a `rules.mk` file in that folder.
 | |
| 3. Include those files from the root folder `testlist.mk`and `build_test.mk` respectively.
 | |
| 4. Add a new name for your testgroup to the `testlist.mk` file. Each group defined there will be a separate executable. And that's how you can support mocking out different parts. Note that it's worth adding some common prefix, just like it's done for the existing tests. The reason for that is that the make command allows substring filtering, so this way you can easily run a subset of the tests.
 | |
| 5. Define the source files and required options in the `rules.mk` file.
 | |
|    * `_SRC` for source files
 | |
|    * `_DEFS` for additional defines
 | |
|    * `_INC` for additional include folders
 | |
| 6. Write the tests in a new cpp file inside the test folder you created. That file has to be one of the files included from the `rules.mk` file.
 | |
| 
 | |
| Note how there's several different tests, each mocking out a separate part. Also note that each of them only compiles the very minimum that's needed for the tests. It's recommend that you try to do the same. For a relevant video check out [Matt Hargett "Advanced Unit Testing in C & C++](https://www.youtube.com/watch?v=Wmy6g-aVgZI)
 | |
| 
 | |
| ## Running the Tests
 | |
| 
 | |
| To run all the tests in the codebase, type `make test:all`. You can also run test matching a substring by typing `make test:matchingsubstring`. `matchingsubstring` can contain colons to be more specific; `make test:tap_hold_configurations` will run the `tap_hold_configurations` tests for all features while `make test:retro_shift:tap_hold_configurations` will run the `tap_hold_configurations` tests for only the Retro Shift feature.
 | |
| 
 | |
| Note that the tests are always compiled with the native compiler of your platform, so they are also run like any other program on your computer.
 | |
| 
 | |
| ## Debugging the Tests
 | |
| 
 | |
| If there are problems with the tests, you can find the executable in the `./build/test` folder. You should be able to run those with GDB or a similar debugger.
 | |
| 
 | |
| To forward any [debug messages](unit_testing#debug-api) to `stderr`, the tests can run with `DEBUG=1`. For example
 | |
| 
 | |
| ```
 | |
| make test:all DEBUG=1
 | |
| ```
 | |
| 
 | |
| Alternatively, add `CONSOLE_ENABLE=yes` to the tests `rules.mk`.
 | |
| 
 | |
| ## Full Integration Tests
 | |
| 
 | |
| It's not yet possible to do a full integration test, where you would compile the whole firmware and define a keymap that you are going to test. However there are plans for doing that, because writing tests that way would probably be easier, at least for people that are not used to unit testing.
 | |
| 
 | |
| In that model you would emulate the input, and expect a certain output from the emulated keyboard.
 | |
| 
 | |
| # Keycode String {#keycode-string}
 | |
| 
 | |
| It's much nicer to read keycodes as names like "`LT(2,KC_D)`" than numerical codes like "`0x4207`." To convert keycodes to human-readable strings, add `KEYCODE_STRING_ENABLE = yes` to the `rules.mk` file, then use the `get_keycode_string(kc)` function to convert a given 16-bit keycode to a string.
 | |
| 
 | |
| ```c
 | |
| const char *key_name = get_keycode_string(keycode);
 | |
| dprintf("kc: %s\n", key_name);
 | |
| ```
 | |
| 
 | |
| The stringified keycode may then be logged to console output with `dprintf()` or elsewhere.
 | |
| 
 | |
| ::: warning
 | |
| Use the result of `get_keycode_string()` immediately. Subsequent invocations reuse the same static buffer and overwrite the previous contents. 
 | |
| :::
 | |
| 
 | |
| Many common QMK keycodes are recognized by `get_keycode_string()`, but not all. These include some common basic keycodes, layer switch keycodes, mod-taps, one-shot keycodes, tap dance keycodes, and Unicode keycodes. As a fallback, an unrecognized keycode is written as a hex number. 
 | |
| 
 | |
| Optionally, `KEYCODE_STRING_NAMES_USER` may be defined to add names for additional keycodes. For example, supposing keymap.c defines `MYMACRO1` and `MYMACRO2` as custom keycodes, the following adds their names:
 | |
| 
 | |
| ```c
 | |
| KEYCODE_STRING_NAMES_USER(
 | |
|     KEYCODE_STRING_NAME(MYMACRO1),
 | |
|     KEYCODE_STRING_NAME(MYMACRO2),
 | |
| );
 | |
| ```
 | |
| 
 | |
| Similarly, `KEYCODE_STRING_NAMES_KB` may be defined to add names at the keyboard level.
 | |
| 
 | |
| # Tracing Variables {#tracing-variables}
 | |
| 
 | |
| Sometimes you might wonder why a variable gets changed and where, and this can be quite tricky to track down without having a debugger. It's of course possible to manually add print statements to track it, but you can also enable the variable trace feature. This works for both variables that are changed by the code, and when the variable is changed by some memory corruption.
 | |
| 
 | |
| To take the feature into use add `VARIABLE_TRACE=x` to the end of you make command. `x` represents the number of variables you want to trace, which is usually 1.
 | |
| 
 | |
| Then at a suitable place in the code, call `ADD_TRACED_VARIABLE`, to begin the tracing. For example to trace all the layer changes, you can do this
 | |
| ```c
 | |
| void matrix_init_user(void) {
 | |
|   ADD_TRACED_VARIABLE("layer", &layer_state, sizeof(layer_state));
 | |
| }
 | |
| ```
 | |
| 
 | |
| This will add a traced variable named "layer" (the name is just for your information), which tracks the memory location of `layer_state`. It tracks 4 bytes (the size of `layer_state`), so any modification to the variable will be reported. By default you can not specify a size bigger than 4, but you can change it by adding `MAX_VARIABLE_TRACE_SIZE=x` to the end of the make command line.
 | |
| 
 | |
| In order to actually detect changes to the variables you should call `VERIFY_TRACED_VARIABLES` around the code that you think that modifies the variable. If a variable is modified it will tell you between which two `VERIFY_TRACED_VARIABLES` calls the modification happened. You can then add more calls to track it down further. I don't recommend spamming the codebase with calls. It's better to start with a few, and then keep adding them in a binary search fashion. You can also delete the ones you don't need, as each call need to store the file name and line number in the ROM, so you can run out of memory if you add too many calls.
 | |
| 
 | |
| Also remember to delete all the tracing code once you have found the bug, as you wouldn't want to create a pull request with tracing code.
 |