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.

913 lines
34 KiB

Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
Add a lot more data to info.json (#13366) * add some split data to info.json * add tags * add half of config_options.md to info.json * add support for designating master split * sort out split transport and primary * fix bad data in UNUSED_PINS * fixup custom transport * wip * allow for setting split right half keyboard matrix * add SPLIT_USB_DETECT * minor cleanup * fix an erroneous message * rework split.usb_detect * adding missing rgblight vars to info.json * add mouse_key to info.json * add all remaining options from docs/config_options.md * fix audio voices * qmk info: Change text output to use dotted notation * tweak layout output * resolve alias names * break out some functions to make flake8 happy * add a field for bootloader instructions * qmk generate-info-json: add a write-to-file argument Adds an argument that instructs qmk generate-info-json to write the output to a file instead of just to the terminal. * -arg_only, +action Because it was never my intention that one would have to specify a value for the argument that enables writing the file. * Bring qmk generate-info-json inline with other generate commands * pytest fixup * fix esca/getawayvan * fix data driven errors for bpiphany converters * features.force_nkro -> usb.force_nkro * split.primary->split.main * fix esca/getawayvan_f042 * fix the bpiphany converters for real * fix bpiphany/tiger_lily * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> * fix generate-api errors * fix matrix pin extraction for split boards * fix ploopyco/trackball_nano/rev1_001 Co-authored-by: James Young <18669334+noroadsleft@users.noreply.github.com> Co-authored-by: Nick Brassel <nick@tzarc.org>
2 years ago
  1. """Functions that help us generate and use info.json files.
  2. """
  3. import re
  4. from pathlib import Path
  5. import jsonschema
  6. from dotty_dict import dotty
  7. from milc import cli
  8. from qmk.constants import CHIBIOS_PROCESSORS, LUFA_PROCESSORS, VUSB_PROCESSORS
  9. from qmk.c_parse import find_layouts, parse_config_h_file, find_led_config
  10. from qmk.json_schema import deep_update, json_load, validate
  11. from qmk.keyboard import config_h, rules_mk
  12. from qmk.commands import parse_configurator_json
  13. from qmk.makefile import parse_rules_mk_file
  14. from qmk.math import compute
  15. true_values = ['1', 'on', 'yes']
  16. false_values = ['0', 'off', 'no']
  17. def _keyboard_in_layout_name(keyboard, layout):
  18. """Validate that a layout macro does not contain name of keyboard
  19. """
  20. # TODO: reduce this list down
  21. safe_layout_tokens = {
  22. 'ansi',
  23. 'iso',
  24. 'jp',
  25. 'jis',
  26. 'ortho',
  27. 'wkl',
  28. 'tkl',
  29. 'preonic',
  30. 'planck',
  31. }
  32. # Ignore tokens like 'split_3x7_4' or just '2x4'
  33. layout = re.sub(r"_split_\d+x\d+_\d+", '', layout)
  34. layout = re.sub(r"_\d+x\d+", '', layout)
  35. name_fragments = set(keyboard.split('/')) - safe_layout_tokens
  36. return any(fragment in layout for fragment in name_fragments)
  37. def _valid_community_layout(layout):
  38. """Validate that a declared community list exists
  39. """
  40. return (Path('layouts/default') / layout).exists()
  41. def _get_key_left_position(key):
  42. # Special case for ISO enter
  43. return key['x'] - 0.25 if key.get('h', 1) == 2 and key.get('w', 1) == 1.25 else key['x']
  44. def _additional_validation(keyboard, info_data):
  45. """Non schema checks
  46. """
  47. layouts = info_data.get('layouts', {})
  48. layout_aliases = info_data.get('layout_aliases', {})
  49. community_layouts = info_data.get('community_layouts', [])
  50. community_layouts_names = list(map(lambda layout: f'LAYOUT_{layout}', community_layouts))
  51. # Make sure we have at least one layout
  52. if len(layouts) == 0 or all(not layout.get('json_layout', False) for layout in layouts.values()):
  53. _log_error(info_data, 'No LAYOUTs defined! Need at least one layout defined in info.json.')
  54. # Warn if physical positions are offset (at least one key should be at x=0, and at least one key at y=0)
  55. for layout_name, layout_data in layouts.items():
  56. offset_x = min([_get_key_left_position(k) for k in layout_data['layout']])
  57. if offset_x > 0:
  58. _log_warning(info_data, f'Layout "{layout_name}" is offset on X axis by {offset_x}')
  59. offset_y = min([k['y'] for k in layout_data['layout']])
  60. if offset_y > 0:
  61. _log_warning(info_data, f'Layout "{layout_name}" is offset on Y axis by {offset_y}')
  62. # Providing only LAYOUT_all "because I define my layouts in a 3rd party tool"
  63. if len(layouts) == 1 and 'LAYOUT_all' in layouts:
  64. _log_warning(info_data, '"LAYOUT_all" should be "LAYOUT" unless additional layouts are provided.')
  65. # Extended layout name checks - ignoring community_layouts and "safe" values
  66. potential_layouts = set(layouts.keys()) - set(community_layouts_names)
  67. for layout in potential_layouts:
  68. if _keyboard_in_layout_name(keyboard, layout):
  69. _log_warning(info_data, f'Layout "{layout}" should not contain name of keyboard.')
  70. # Filter out any non-existing community layouts
  71. for layout in community_layouts:
  72. if not _valid_community_layout(layout):
  73. # Ignore layout from future checks
  74. info_data['community_layouts'].remove(layout)
  75. _log_error(info_data, 'Claims to support a community layout that does not exist: %s' % (layout))
  76. # Make sure we supply layout macros for the community layouts we claim to support
  77. for layout_name in community_layouts_names:
  78. if layout_name not in layouts and layout_name not in layout_aliases:
  79. _log_error(info_data, 'Claims to support community layout %s but no %s() macro found' % (layout, layout_name))
  80. # keycodes with length > 7 must have short forms for visualisation purposes
  81. for decl in info_data.get('keycodes', []):
  82. if len(decl["key"]) > 7:
  83. if not decl.get("aliases", []):
  84. _log_error(info_data, f'Keycode {decl["key"]} has no short form alias')
  85. def _validate(keyboard, info_data):
  86. """Perform various validation on the provided info.json data
  87. """
  88. # First validate against the jsonschema
  89. try:
  90. validate(info_data, 'qmk.api.keyboard.v1')
  91. _additional_validation(keyboard, info_data)
  92. except jsonschema.ValidationError as e:
  93. json_path = '.'.join([str(p) for p in e.absolute_path])
  94. cli.log.error('Invalid API data: %s: %s: %s', keyboard, json_path, e.message)
  95. exit(1)
  96. def info_json(keyboard):
  97. """Generate the info.json data for a specific keyboard.
  98. """
  99. cur_dir = Path('keyboards')
  100. root_rules_mk = parse_rules_mk_file(cur_dir / keyboard / 'rules.mk')
  101. if 'DEFAULT_FOLDER' in root_rules_mk:
  102. keyboard = root_rules_mk['DEFAULT_FOLDER']
  103. info_data = {
  104. 'keyboard_name': str(keyboard),
  105. 'keyboard_folder': str(keyboard),
  106. 'keymaps': {},
  107. 'layouts': {},
  108. 'parse_errors': [],
  109. 'parse_warnings': [],
  110. 'maintainer': 'qmk',
  111. }
  112. # Populate layout data
  113. layouts, aliases = _search_keyboard_h(keyboard)
  114. if aliases:
  115. info_data['layout_aliases'] = aliases
  116. for layout_name, layout_json in layouts.items():
  117. if not layout_name.startswith('LAYOUT_kc'):
  118. layout_json['c_macro'] = True
  119. layout_json['json_layout'] = False
  120. info_data['layouts'][layout_name] = layout_json
  121. # Merge in the data from info.json, config.h, and rules.mk
  122. info_data = merge_info_jsons(keyboard, info_data)
  123. info_data = _process_defaults(info_data)
  124. info_data = _extract_rules_mk(info_data, rules_mk(str(keyboard)))
  125. info_data = _extract_config_h(info_data, config_h(str(keyboard)))
  126. # Ensure that we have matrix row and column counts
  127. info_data = _matrix_size(info_data)
  128. # Merge in data from <keyboard.c>
  129. info_data = _extract_led_config(info_data, str(keyboard))
  130. # Validate
  131. _validate(keyboard, info_data)
  132. # Check that the reported matrix size is consistent with the actual matrix size
  133. _check_matrix(info_data)
  134. return info_data
  135. def _extract_features(info_data, rules):
  136. """Find all the features enabled in rules.mk.
  137. """
  138. # Process booleans rules
  139. for key, value in rules.items():
  140. if key.endswith('_ENABLE'):
  141. key = '_'.join(key.split('_')[:-1]).lower()
  142. value = True if value.lower() in true_values else False if value.lower() in false_values else value
  143. if 'config_h_features' not in info_data:
  144. info_data['config_h_features'] = {}
  145. if 'features' not in info_data:
  146. info_data['features'] = {}
  147. if key in info_data['features']:
  148. _log_warning(info_data, 'Feature %s is specified in both info.json and rules.mk, the rules.mk value wins.' % (key,))
  149. info_data['features'][key] = value
  150. info_data['config_h_features'][key] = value
  151. return info_data
  152. def _pin_name(pin):
  153. """Returns the proper representation for a pin.
  154. """
  155. pin = pin.strip()
  156. if not pin:
  157. return None
  158. elif pin.isdigit():
  159. return int(pin)
  160. elif pin == 'NO_PIN':
  161. return None
  162. return pin
  163. def _extract_pins(pins):
  164. """Returns a list of pins from a comma separated string of pins.
  165. """
  166. return [_pin_name(pin) for pin in pins.split(',')]
  167. def _extract_2d_array(raw):
  168. """Return a 2d array of strings
  169. """
  170. out_array = []
  171. while raw[-1] != '}':
  172. raw = raw[:-1]
  173. for row in raw.split('},{'):
  174. if row.startswith('{'):
  175. row = row[1:]
  176. if row.endswith('}'):
  177. row = row[:-1]
  178. out_array.append([])
  179. for val in row.split(','):
  180. out_array[-1].append(val)
  181. return out_array
  182. def _extract_2d_int_array(raw):
  183. """Return a 2d array of ints
  184. """
  185. ret = _extract_2d_array(raw)
  186. return [list(map(int, x)) for x in ret]
  187. def _extract_direct_matrix(direct_pins):
  188. """extract direct_matrix
  189. """
  190. direct_pin_array = _extract_2d_array(direct_pins)
  191. for i in range(len(direct_pin_array)):
  192. for j in range(len(direct_pin_array[i])):
  193. if direct_pin_array[i][j] == 'NO_PIN':
  194. direct_pin_array[i][j] = None
  195. return direct_pin_array
  196. def _extract_audio(info_data, config_c):
  197. """Populate data about the audio configuration
  198. """
  199. audio_pins = []
  200. for pin in 'B5', 'B6', 'B7', 'C4', 'C5', 'C6':
  201. if config_c.get(f'{pin}_AUDIO'):
  202. audio_pins.append(pin)
  203. if audio_pins:
  204. info_data['audio'] = {'pins': audio_pins}
  205. def _extract_encoders_values(config_c, postfix=''):
  206. """Common encoder extraction logic
  207. """
  208. a_pad = config_c.get(f'ENCODERS_PAD_A{postfix}', '').replace(' ', '')[1:-1]
  209. b_pad = config_c.get(f'ENCODERS_PAD_B{postfix}', '').replace(' ', '')[1:-1]
  210. resolutions = config_c.get(f'ENCODER_RESOLUTIONS{postfix}', '').replace(' ', '')[1:-1]
  211. default_resolution = config_c.get('ENCODER_RESOLUTION', None)
  212. if a_pad and b_pad:
  213. a_pad = list(filter(None, a_pad.split(',')))
  214. b_pad = list(filter(None, b_pad.split(',')))
  215. resolutions = list(filter(None, resolutions.split(',')))
  216. if default_resolution:
  217. resolutions += [default_resolution] * (len(a_pad) - len(resolutions))
  218. encoders = []
  219. for index in range(len(a_pad)):
  220. encoder = {'pin_a': a_pad[index], 'pin_b': b_pad[index]}
  221. if index < len(resolutions):
  222. encoder['resolution'] = int(resolutions[index])
  223. encoders.append(encoder)
  224. return encoders
  225. def _extract_encoders(info_data, config_c):
  226. """Populate data about encoder pins
  227. """
  228. encoders = _extract_encoders_values(config_c)
  229. if encoders:
  230. if 'encoder' not in info_data:
  231. info_data['encoder'] = {}
  232. if 'rotary' in info_data['encoder']:
  233. _log_warning(info_data, 'Encoder config is specified in both config.h and info.json (encoder.rotary) (Value: %s), the config.h value wins.' % info_data['encoder']['rotary'])
  234. info_data['encoder']['rotary'] = encoders
  235. def _extract_split_encoders(info_data, config_c):
  236. """Populate data about split encoder pins
  237. """
  238. encoders = _extract_encoders_values(config_c, '_RIGHT')
  239. if encoders:
  240. if 'split' not in info_data:
  241. info_data['split'] = {}
  242. if 'encoder' not in info_data['split']:
  243. info_data['split']['encoder'] = {}
  244. if 'right' not in info_data['split']['encoder']:
  245. info_data['split']['encoder']['right'] = {}
  246. if 'rotary' in info_data['split']['encoder']['right']:
  247. _log_warning(info_data, 'Encoder config is specified in both config.h and info.json (encoder.rotary) (Value: %s), the config.h value wins.' % info_data['split']['encoder']['right']['rotary'])
  248. info_data['split']['encoder']['right']['rotary'] = encoders
  249. def _extract_secure_unlock(info_data, config_c):
  250. """Populate data about the secure unlock sequence
  251. """
  252. unlock = config_c.get('SECURE_UNLOCK_SEQUENCE', '').replace(' ', '')[1:-1]
  253. if unlock:
  254. unlock_array = _extract_2d_int_array(unlock)
  255. if 'secure' not in info_data:
  256. info_data['secure'] = {}
  257. if 'unlock_sequence' in info_data['secure']:
  258. _log_warning(info_data, 'Secure unlock sequence is specified in both config.h (SECURE_UNLOCK_SEQUENCE) and info.json (secure.unlock_sequence) (Value: %s), the config.h value wins.' % info_data['secure']['unlock_sequence'])
  259. info_data['secure']['unlock_sequence'] = unlock_array
  260. def _extract_split_transport(info_data, config_c):
  261. # Figure out the transport method
  262. if config_c.get('USE_I2C') is True:
  263. if 'split' not in info_data:
  264. info_data['split'] = {}
  265. if 'transport' not in info_data['split']:
  266. info_data['split']['transport'] = {}
  267. if 'protocol' in info_data['split']['transport']:
  268. _log_warning(info_data, 'Split transport is specified in both config.h (USE_I2C) and info.json (split.transport.protocol) (Value: %s), the config.h value wins.' % info_data['split']['transport'])
  269. info_data['split']['transport']['protocol'] = 'i2c'
  270. # Ignore transport defaults if "SPLIT_KEYBOARD" is unset
  271. elif 'enabled' in info_data.get('split', {}):
  272. if 'split' not in info_data:
  273. info_data['split'] = {}
  274. if 'transport' not in info_data['split']:
  275. info_data['split']['transport'] = {}
  276. if 'protocol' not in info_data['split']['transport']:
  277. info_data['split']['transport']['protocol'] = 'serial'
  278. # Migrate
  279. transport = info_data.get('split', {}).get('transport', {})
  280. if 'sync_matrix_state' in transport:
  281. transport['sync'] = transport.get('sync', {})
  282. transport['sync']['matrix_state'] = transport.pop('sync_matrix_state')
  283. if 'sync_modifiers' in transport:
  284. transport['sync'] = transport.get('sync', {})
  285. transport['sync']['modifiers'] = transport.pop('sync_modifiers')
  286. def _extract_split_right_pins(info_data, config_c):
  287. # Figure out the right half matrix pins
  288. row_pins = config_c.get('MATRIX_ROW_PINS_RIGHT', '').replace('{', '').replace('}', '').strip()
  289. col_pins = config_c.get('MATRIX_COL_PINS_RIGHT', '').replace('{', '').replace('}', '').strip()
  290. direct_pins = config_c.get('DIRECT_PINS_RIGHT', '').replace(' ', '')[1:-1]
  291. if row_pins or col_pins or direct_pins:
  292. if info_data.get('split', {}).get('matrix_pins', {}).get('right', None):
  293. _log_warning(info_data, 'Right hand matrix data is specified in both info.json and config.h, the config.h values win.')
  294. if 'split' not in info_data:
  295. info_data['split'] = {}
  296. if 'matrix_pins' not in info_data['split']:
  297. info_data['split']['matrix_pins'] = {}
  298. if 'right' not in info_data['split']['matrix_pins']:
  299. info_data['split']['matrix_pins']['right'] = {}
  300. if col_pins:
  301. info_data['split']['matrix_pins']['right']['cols'] = _extract_pins(col_pins)
  302. if row_pins:
  303. info_data['split']['matrix_pins']['right']['rows'] = _extract_pins(row_pins)
  304. if direct_pins:
  305. info_data['split']['matrix_pins']['right']['direct'] = _extract_direct_matrix(direct_pins)
  306. def _extract_matrix_info(info_data, config_c):
  307. """Populate the matrix information.
  308. """
  309. row_pins = config_c.get('MATRIX_ROW_PINS', '').replace('{', '').replace('}', '').strip()
  310. col_pins = config_c.get('MATRIX_COL_PINS', '').replace('{', '').replace('}', '').strip()
  311. direct_pins = config_c.get('DIRECT_PINS', '').replace(' ', '')[1:-1]
  312. info_snippet = {}
  313. if 'MATRIX_ROWS' in config_c and 'MATRIX_COLS' in config_c:
  314. if 'matrix_size' in info_data:
  315. _log_warning(info_data, 'Matrix size is specified in both info.json and config.h, the config.h values win.')
  316. info_data['matrix_size'] = {
  317. 'cols': compute(config_c.get('MATRIX_COLS', '0')),
  318. 'rows': compute(config_c.get('MATRIX_ROWS', '0')),
  319. }
  320. if row_pins and col_pins:
  321. if 'matrix_pins' in info_data and 'cols' in info_data['matrix_pins'] and 'rows' in info_data['matrix_pins']:
  322. _log_warning(info_data, 'Matrix pins are specified in both info.json and config.h, the config.h values win.')
  323. info_snippet['cols'] = _extract_pins(col_pins)
  324. info_snippet['rows'] = _extract_pins(row_pins)
  325. if direct_pins:
  326. if 'matrix_pins' in info_data and 'direct' in info_data['matrix_pins']:
  327. _log_warning(info_data, 'Direct pins are specified in both info.json and config.h, the config.h values win.')
  328. info_snippet['direct'] = _extract_direct_matrix(direct_pins)
  329. if config_c.get('CUSTOM_MATRIX', 'no') != 'no':
  330. if 'matrix_pins' in info_data and 'custom' in info_data['matrix_pins']:
  331. _log_warning(info_data, 'Custom Matrix is specified in both info.json and config.h, the config.h values win.')
  332. info_snippet['custom'] = True
  333. if config_c['CUSTOM_MATRIX'] == 'lite':
  334. info_snippet['custom_lite'] = True
  335. if info_snippet:
  336. info_data['matrix_pins'] = info_snippet
  337. return info_data
  338. def _config_to_json(key_type, config_value):
  339. """Convert config value using spec
  340. """
  341. if key_type.startswith('array'):
  342. if '.' in key_type:
  343. key_type, array_type = key_type.split('.', 1)
  344. else:
  345. array_type = None
  346. config_value = config_value.replace('{', '').replace('}', '').strip()
  347. if array_type == 'int':
  348. return list(map(int, config_value.split(',')))
  349. else:
  350. return list(map(str.strip, config_value.split(',')))
  351. elif key_type == 'bool':
  352. if isinstance(config_value, bool):
  353. return config_value
  354. return config_value in true_values
  355. elif key_type == 'hex':
  356. return '0x' + config_value[2:].upper()
  357. elif key_type == 'list':
  358. return config_value.split()
  359. elif key_type == 'int':
  360. return int(config_value)
  361. elif key_type == 'str':
  362. return config_value.strip('"').replace('\\"', '"').replace('\\\\', '\\')
  363. elif key_type == 'bcd_version':
  364. major = int(config_value[2:4])
  365. minor = int(config_value[4])
  366. revision = int(config_value[5])
  367. return f'{major}.{minor}.{revision}'
  368. return config_value
  369. def _extract_config_h(info_data, config_c):
  370. """Pull some keyboard information from existing config.h files
  371. """
  372. # Pull in data from the json map
  373. dotty_info = dotty(info_data)
  374. info_config_map = json_load(Path('data/mappings/info_config.hjson'))
  375. for config_key, info_dict in info_config_map.items():
  376. info_key = info_dict['info_key']
  377. key_type = info_dict.get('value_type', 'raw')
  378. try:
  379. replace_with = info_dict.get('replace_with')
  380. if config_key in config_c and info_dict.get('invalid', False):
  381. if replace_with:
  382. _log_error(info_data, '%s in config.h is no longer a valid option and should be replaced with %s' % (config_key, replace_with))
  383. else:
  384. _log_error(info_data, '%s in config.h is no longer a valid option and should be removed' % config_key)
  385. elif config_key in config_c and info_dict.get('deprecated', False):
  386. if replace_with:
  387. _log_warning(info_data, '%s in config.h is deprecated in favor of %s and will be removed at a later date' % (config_key, replace_with))
  388. else:
  389. _log_warning(info_data, '%s in config.h is deprecated and will be removed at a later date' % config_key)
  390. if config_key in config_c and info_dict.get('to_json', True):
  391. if dotty_info.get(info_key) and info_dict.get('warn_duplicate', True):
  392. _log_warning(info_data, '%s in config.h is overwriting %s in info.json' % (config_key, info_key))
  393. dotty_info[info_key] = _config_to_json(key_type, config_c[config_key])
  394. except Exception as e:
  395. _log_warning(info_data, f'{config_key}->{info_key}: {e}')
  396. info_data.update(dotty_info)
  397. # Pull data that easily can't be mapped in json
  398. _extract_matrix_info(info_data, config_c)
  399. _extract_audio(info_data, config_c)
  400. _extract_secure_unlock(info_data, config_c)
  401. _extract_split_transport(info_data, config_c)
  402. _extract_split_right_pins(info_data, config_c)
  403. _extract_encoders(info_data, config_c)
  404. _extract_split_encoders(info_data, config_c)
  405. return info_data
  406. def _process_defaults(info_data):
  407. """Process any additional defaults based on currently discovered information
  408. """
  409. defaults_map = json_load(Path('data/mappings/defaults.hjson'))
  410. for default_type in defaults_map.keys():
  411. thing_map = defaults_map[default_type]
  412. if default_type in info_data:
  413. merged_count = 0
  414. thing_items = thing_map.get(info_data[default_type], {}).items()
  415. for key, value in thing_items:
  416. if key not in info_data:
  417. info_data[key] = value
  418. merged_count += 1
  419. if merged_count == 0 and len(thing_items) > 0:
  420. _log_warning(info_data, 'All defaults for \'%s\' were skipped, potential redundant config or misconfiguration detected' % (default_type))
  421. return info_data
  422. def _extract_rules_mk(info_data, rules):
  423. """Pull some keyboard information from existing rules.mk files
  424. """
  425. info_data['processor'] = rules.get('MCU', info_data.get('processor', 'atmega32u4'))
  426. if info_data['processor'] in CHIBIOS_PROCESSORS:
  427. arm_processor_rules(info_data, rules)
  428. elif info_data['processor'] in LUFA_PROCESSORS + VUSB_PROCESSORS:
  429. avr_processor_rules(info_data, rules)
  430. else:
  431. cli.log.warning("%s: Unknown MCU: %s" % (info_data['keyboard_folder'], info_data['processor']))
  432. unknown_processor_rules(info_data, rules)
  433. # Pull in data from the json map
  434. dotty_info = dotty(info_data)
  435. info_rules_map = json_load(Path('data/mappings/info_rules.hjson'))
  436. for rules_key, info_dict in info_rules_map.items():
  437. info_key = info_dict['info_key']
  438. key_type = info_dict.get('value_type', 'raw')
  439. try:
  440. replace_with = info_dict.get('replace_with')
  441. if rules_key in rules and info_dict.get('invalid', False):
  442. if replace_with:
  443. _log_error(info_data, '%s in rules.mk is no longer a valid option and should be replaced with %s' % (rules_key, replace_with))
  444. else:
  445. _log_error(info_data, '%s in rules.mk is no longer a valid option and should be removed' % rules_key)
  446. elif rules_key in rules and info_dict.get('deprecated', False):
  447. if replace_with:
  448. _log_warning(info_data, '%s in rules.mk is deprecated in favor of %s and will be removed at a later date' % (rules_key, replace_with))
  449. else:
  450. _log_warning(info_data, '%s in rules.mk is deprecated and will be removed at a later date' % rules_key)
  451. if rules_key in rules and info_dict.get('to_json', True):
  452. if dotty_info.get(info_key) and info_dict.get('warn_duplicate', True):
  453. _log_warning(info_data, '%s in rules.mk is overwriting %s in info.json' % (rules_key, info_key))
  454. dotty_info[info_key] = _config_to_json(key_type, rules[rules_key])
  455. except Exception as e:
  456. _log_warning(info_data, f'{rules_key}->{info_key}: {e}')
  457. info_data.update(dotty_info)
  458. # Merge in config values that can't be easily mapped
  459. _extract_features(info_data, rules)
  460. return info_data
  461. def find_keyboard_c(keyboard):
  462. """Find all <keyboard>.c files
  463. """
  464. keyboard = Path(keyboard)
  465. current_path = Path('keyboards/')
  466. files = []
  467. for directory in keyboard.parts:
  468. current_path = current_path / directory
  469. keyboard_c_path = current_path / f'{directory}.c'
  470. if keyboard_c_path.exists():
  471. files.append(keyboard_c_path)
  472. return files
  473. def _extract_led_config(info_data, keyboard):
  474. """Scan all <keyboard>.c files for led config
  475. """
  476. cols = info_data['matrix_size']['cols']
  477. rows = info_data['matrix_size']['rows']
  478. # Determine what feature owns g_led_config
  479. features = info_data.get("features", {})
  480. feature = None
  481. if features.get("rgb_matrix", False):
  482. feature = "rgb_matrix"
  483. elif features.get("led_matrix", False):
  484. feature = "led_matrix"
  485. if feature:
  486. # Process
  487. for file in find_keyboard_c(keyboard):
  488. try:
  489. ret = find_led_config(file, cols, rows)
  490. if ret:
  491. info_data[feature] = info_data.get(feature, {})
  492. info_data[feature]["layout"] = ret
  493. except Exception as e:
  494. _log_warning(info_data, f'led_config: {file.name}: {e}')
  495. if info_data[feature].get("layout", None) and not info_data[feature].get("led_count", None):
  496. info_data[feature]["led_count"] = len(info_data[feature]["layout"])
  497. return info_data
  498. def _matrix_size(info_data):
  499. """Add info_data['matrix_size'] if it doesn't exist.
  500. """
  501. if 'matrix_size' not in info_data and 'matrix_pins' in info_data:
  502. info_data['matrix_size'] = {}
  503. if 'direct' in info_data['matrix_pins']:
  504. info_data['matrix_size']['cols'] = len(info_data['matrix_pins']['direct'][0])
  505. info_data['matrix_size']['rows'] = len(info_data['matrix_pins']['direct'])
  506. elif 'cols' in info_data['matrix_pins'] and 'rows' in info_data['matrix_pins']:
  507. info_data['matrix_size']['cols'] = len(info_data['matrix_pins']['cols'])
  508. info_data['matrix_size']['rows'] = len(info_data['matrix_pins']['rows'])
  509. # Assumption of split common
  510. if 'split' in info_data:
  511. if info_data['split'].get('enabled', False):
  512. info_data['matrix_size']['rows'] *= 2
  513. return info_data
  514. def _check_matrix(info_data):
  515. """Check the matrix to ensure that row/column count is consistent.
  516. """
  517. if 'matrix_pins' in info_data and 'matrix_size' in info_data:
  518. actual_col_count = info_data['matrix_size'].get('cols', 0)
  519. actual_row_count = info_data['matrix_size'].get('rows', 0)
  520. col_count = row_count = 0
  521. if 'direct' in info_data['matrix_pins']:
  522. col_count = len(info_data['matrix_pins']['direct'][0])
  523. row_count = len(info_data['matrix_pins']['direct'])
  524. elif 'cols' in info_data['matrix_pins'] and 'rows' in info_data['matrix_pins']:
  525. col_count = len(info_data['matrix_pins']['cols'])
  526. row_count = len(info_data['matrix_pins']['rows'])
  527. if col_count != actual_col_count and col_count != (actual_col_count / 2):
  528. # FIXME: once we can we should detect if split is enabled to do the actual_col_count/2 check.
  529. _log_error(info_data, f'MATRIX_COLS is inconsistent with the size of MATRIX_COL_PINS: {col_count} != {actual_col_count}')
  530. if row_count != actual_row_count and row_count != (actual_row_count / 2):
  531. # FIXME: once we can we should detect if split is enabled to do the actual_row_count/2 check.
  532. _log_error(info_data, f'MATRIX_ROWS is inconsistent with the size of MATRIX_ROW_PINS: {row_count} != {actual_row_count}')
  533. def _search_keyboard_h(keyboard):
  534. keyboard = Path(keyboard)
  535. current_path = Path('keyboards/')
  536. aliases = {}
  537. layouts = {}
  538. for directory in keyboard.parts:
  539. current_path = current_path / directory
  540. keyboard_h = '%s.h' % (directory,)
  541. keyboard_h_path = current_path / keyboard_h
  542. if keyboard_h_path.exists():
  543. new_layouts, new_aliases = find_layouts(keyboard_h_path)
  544. layouts.update(new_layouts)
  545. for alias, alias_text in new_aliases.items():
  546. if alias_text in layouts:
  547. aliases[alias] = alias_text
  548. return layouts, aliases
  549. def _log_error(info_data, message):
  550. """Send an error message to both JSON and the log.
  551. """
  552. info_data['parse_errors'].append(message)
  553. cli.log.error('%s: %s', info_data.get('keyboard_folder', 'Unknown Keyboard!'), message)
  554. def _log_warning(info_data, message):
  555. """Send a warning message to both JSON and the log.
  556. """
  557. info_data['parse_warnings'].append(message)
  558. cli.log.warning('%s: %s', info_data.get('keyboard_folder', 'Unknown Keyboard!'), message)
  559. def arm_processor_rules(info_data, rules):
  560. """Setup the default info for an ARM board.
  561. """
  562. info_data['processor_type'] = 'arm'
  563. info_data['protocol'] = 'ChibiOS'
  564. info_data['platform_key'] = 'chibios'
  565. if 'STM32' in info_data['processor']:
  566. info_data['platform'] = 'STM32'
  567. elif 'MCU_SERIES' in rules:
  568. info_data['platform'] = rules['MCU_SERIES']
  569. elif 'ARM_ATSAM' in rules:
  570. info_data['platform'] = 'ARM_ATSAM'
  571. info_data['platform_key'] = 'arm_atsam'
  572. return info_data
  573. def avr_processor_rules(info_data, rules):
  574. """Setup the default info for an AVR board.
  575. """
  576. info_data['processor_type'] = 'avr'
  577. info_data['platform'] = rules['ARCH'] if 'ARCH' in rules else 'unknown'
  578. info_data['platform_key'] = 'avr'
  579. info_data['protocol'] = 'V-USB' if info_data['processor'] in VUSB_PROCESSORS else 'LUFA'
  580. # FIXME(fauxpark/anyone): Eventually we should detect the protocol by looking at PROTOCOL inherited from mcu_selection.mk:
  581. # info_data['protocol'] = 'V-USB' if rules.get('PROTOCOL') == 'VUSB' else 'LUFA'
  582. return info_data
  583. def unknown_processor_rules(info_data, rules):
  584. """Setup the default keyboard info for unknown boards.
  585. """
  586. info_data['bootloader'] = 'unknown'
  587. info_data['platform'] = 'unknown'
  588. info_data['processor'] = 'unknown'
  589. info_data['processor_type'] = 'unknown'
  590. info_data['protocol'] = 'unknown'
  591. return info_data
  592. def merge_info_jsons(keyboard, info_data):
  593. """Return a merged copy of all the info.json files for a keyboard.
  594. """
  595. for info_file in find_info_json(keyboard):
  596. # Load and validate the JSON data
  597. new_info_data = json_load(info_file)
  598. if not isinstance(new_info_data, dict):
  599. _log_error(info_data, "Invalid file %s, root object should be a dictionary." % (str(info_file),))
  600. continue
  601. try:
  602. validate(new_info_data, 'qmk.keyboard.v1')
  603. except jsonschema.ValidationError as e:
  604. json_path = '.'.join([str(p) for p in e.absolute_path])
  605. cli.log.error('Not including data from file: %s', info_file)
  606. cli.log.error('\t%s: %s', json_path, e.message)
  607. continue
  608. # Merge layout data in
  609. if 'layout_aliases' in new_info_data:
  610. info_data['layout_aliases'] = {**info_data.get('layout_aliases', {}), **new_info_data['layout_aliases']}
  611. del new_info_data['layout_aliases']
  612. for layout_name, layout in new_info_data.get('layouts', {}).items():
  613. if layout_name in info_data.get('layout_aliases', {}):
  614. _log_warning(info_data, f"info.json uses alias name {layout_name} instead of {info_data['layout_aliases'][layout_name]}")
  615. layout_name = info_data['layout_aliases'][layout_name]
  616. if layout_name in info_data['layouts']:
  617. if len(info_data['layouts'][layout_name]['layout']) != len(layout['layout']):
  618. msg = 'Number of keys for %s does not match! info.json specifies %d keys, C macro specifies %d'
  619. _log_error(info_data, msg % (layout_name, len(layout['layout']), len(info_data['layouts'][layout_name]['layout'])))
  620. else:
  621. info_data['layouts'][layout_name]['json_layout'] = True
  622. for new_key, existing_key in zip(layout['layout'], info_data['layouts'][layout_name]['layout']):
  623. existing_key.update(new_key)
  624. else:
  625. if not all('matrix' in key_data.keys() for key_data in layout['layout']):
  626. _log_error(info_data, f'Layout "{layout_name}" has no "matrix" definition in either "info.json" or "<keyboard>.h"!')
  627. else:
  628. layout['c_macro'] = False
  629. layout['json_layout'] = True
  630. info_data['layouts'][layout_name] = layout
  631. # Update info_data with the new data
  632. if 'layouts' in new_info_data:
  633. del new_info_data['layouts']
  634. deep_update(info_data, new_info_data)
  635. return info_data
  636. def find_info_json(keyboard):
  637. """Finds all the info.json files associated with a keyboard.
  638. """
  639. # Find the most specific first
  640. base_path = Path('keyboards')
  641. keyboard_path = base_path / keyboard
  642. keyboard_parent = keyboard_path.parent
  643. info_jsons = [keyboard_path / 'info.json']
  644. # Add DEFAULT_FOLDER before parents, if present
  645. rules = rules_mk(keyboard)
  646. if 'DEFAULT_FOLDER' in rules:
  647. info_jsons.append(Path(rules['DEFAULT_FOLDER']) / 'info.json')
  648. # Add in parent folders for least specific
  649. for _ in range(5):
  650. if keyboard_parent == base_path:
  651. break
  652. info_jsons.append(keyboard_parent / 'info.json')
  653. keyboard_parent = keyboard_parent.parent
  654. # Return a list of the info.json files that actually exist
  655. return [info_json for info_json in info_jsons if info_json.exists()]
  656. def keymap_json_config(keyboard, keymap):
  657. """Extract keymap level config
  658. """
  659. # TODO: resolve keymap.py and info.py circular dependencies
  660. from qmk.keymap import locate_keymap
  661. keymap_folder = locate_keymap(keyboard, keymap).parent
  662. km_info_json = parse_configurator_json(keymap_folder / 'keymap.json')
  663. return km_info_json.get('config', {})
  664. def keymap_json(keyboard, keymap):
  665. """Generate the info.json data for a specific keymap.
  666. """
  667. # TODO: resolve keymap.py and info.py circular dependencies
  668. from qmk.keymap import locate_keymap
  669. keymap_folder = locate_keymap(keyboard, keymap).parent
  670. # Files to scan
  671. keymap_config = keymap_folder / 'config.h'
  672. keymap_rules = keymap_folder / 'rules.mk'
  673. keymap_file = keymap_folder / 'keymap.json'
  674. # Build the info.json file
  675. kb_info_json = info_json(keyboard)
  676. # Merge in the data from keymap.json
  677. km_info_json = keymap_json_config(keyboard, keymap) if keymap_file.exists() else {}
  678. deep_update(kb_info_json, km_info_json)
  679. # Merge in the data from config.h, and rules.mk
  680. _extract_rules_mk(kb_info_json, parse_rules_mk_file(keymap_rules))
  681. _extract_config_h(kb_info_json, parse_config_h_file(keymap_config))
  682. return kb_info_json