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.

922 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_handedness(info_data, config_c):
  261. # Migrate
  262. split = info_data.get('split', {})
  263. if 'matrix_grid' in split:
  264. split['handedness'] = split.get('handedness', {})
  265. split['handedness']['matrix_grid'] = split.pop('matrix_grid')
  266. def _extract_split_transport(info_data, config_c):
  267. # Figure out the transport method
  268. if config_c.get('USE_I2C') is True:
  269. if 'split' not in info_data:
  270. info_data['split'] = {}
  271. if 'transport' not in info_data['split']:
  272. info_data['split']['transport'] = {}
  273. if 'protocol' in info_data['split']['transport']:
  274. _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'])
  275. info_data['split']['transport']['protocol'] = 'i2c'
  276. # Ignore transport defaults if "SPLIT_KEYBOARD" is unset
  277. elif 'enabled' in info_data.get('split', {}):
  278. if 'split' not in info_data:
  279. info_data['split'] = {}
  280. if 'transport' not in info_data['split']:
  281. info_data['split']['transport'] = {}
  282. if 'protocol' not in info_data['split']['transport']:
  283. info_data['split']['transport']['protocol'] = 'serial'
  284. # Migrate
  285. transport = info_data.get('split', {}).get('transport', {})
  286. if 'sync_matrix_state' in transport:
  287. transport['sync'] = transport.get('sync', {})
  288. transport['sync']['matrix_state'] = transport.pop('sync_matrix_state')
  289. if 'sync_modifiers' in transport:
  290. transport['sync'] = transport.get('sync', {})
  291. transport['sync']['modifiers'] = transport.pop('sync_modifiers')
  292. def _extract_split_right_pins(info_data, config_c):
  293. # Figure out the right half matrix pins
  294. row_pins = config_c.get('MATRIX_ROW_PINS_RIGHT', '').replace('{', '').replace('}', '').strip()
  295. col_pins = config_c.get('MATRIX_COL_PINS_RIGHT', '').replace('{', '').replace('}', '').strip()
  296. direct_pins = config_c.get('DIRECT_PINS_RIGHT', '').replace(' ', '')[1:-1]
  297. if row_pins or col_pins or direct_pins:
  298. if info_data.get('split', {}).get('matrix_pins', {}).get('right', None):
  299. _log_warning(info_data, 'Right hand matrix data is specified in both info.json and config.h, the config.h values win.')
  300. if 'split' not in info_data:
  301. info_data['split'] = {}
  302. if 'matrix_pins' not in info_data['split']:
  303. info_data['split']['matrix_pins'] = {}
  304. if 'right' not in info_data['split']['matrix_pins']:
  305. info_data['split']['matrix_pins']['right'] = {}
  306. if col_pins:
  307. info_data['split']['matrix_pins']['right']['cols'] = _extract_pins(col_pins)
  308. if row_pins:
  309. info_data['split']['matrix_pins']['right']['rows'] = _extract_pins(row_pins)
  310. if direct_pins:
  311. info_data['split']['matrix_pins']['right']['direct'] = _extract_direct_matrix(direct_pins)
  312. def _extract_matrix_info(info_data, config_c):
  313. """Populate the matrix information.
  314. """
  315. row_pins = config_c.get('MATRIX_ROW_PINS', '').replace('{', '').replace('}', '').strip()
  316. col_pins = config_c.get('MATRIX_COL_PINS', '').replace('{', '').replace('}', '').strip()
  317. direct_pins = config_c.get('DIRECT_PINS', '').replace(' ', '')[1:-1]
  318. info_snippet = {}
  319. if 'MATRIX_ROWS' in config_c and 'MATRIX_COLS' in config_c:
  320. if 'matrix_size' in info_data:
  321. _log_warning(info_data, 'Matrix size is specified in both info.json and config.h, the config.h values win.')
  322. info_data['matrix_size'] = {
  323. 'cols': compute(config_c.get('MATRIX_COLS', '0')),
  324. 'rows': compute(config_c.get('MATRIX_ROWS', '0')),
  325. }
  326. if row_pins and col_pins:
  327. if 'matrix_pins' in info_data and 'cols' in info_data['matrix_pins'] and 'rows' in info_data['matrix_pins']:
  328. _log_warning(info_data, 'Matrix pins are specified in both info.json and config.h, the config.h values win.')
  329. info_snippet['cols'] = _extract_pins(col_pins)
  330. info_snippet['rows'] = _extract_pins(row_pins)
  331. if direct_pins:
  332. if 'matrix_pins' in info_data and 'direct' in info_data['matrix_pins']:
  333. _log_warning(info_data, 'Direct pins are specified in both info.json and config.h, the config.h values win.')
  334. info_snippet['direct'] = _extract_direct_matrix(direct_pins)
  335. if config_c.get('CUSTOM_MATRIX', 'no') != 'no':
  336. if 'matrix_pins' in info_data and 'custom' in info_data['matrix_pins']:
  337. _log_warning(info_data, 'Custom Matrix is specified in both info.json and config.h, the config.h values win.')
  338. info_snippet['custom'] = True
  339. if config_c['CUSTOM_MATRIX'] == 'lite':
  340. info_snippet['custom_lite'] = True
  341. if info_snippet:
  342. info_data['matrix_pins'] = info_snippet
  343. return info_data
  344. def _config_to_json(key_type, config_value):
  345. """Convert config value using spec
  346. """
  347. if key_type.startswith('array'):
  348. if '.' in key_type:
  349. key_type, array_type = key_type.split('.', 1)
  350. else:
  351. array_type = None
  352. config_value = config_value.replace('{', '').replace('}', '').strip()
  353. if array_type == 'int':
  354. return list(map(int, config_value.split(',')))
  355. else:
  356. return list(map(str.strip, config_value.split(',')))
  357. elif key_type == 'bool':
  358. if isinstance(config_value, bool):
  359. return config_value
  360. return config_value in true_values
  361. elif key_type == 'hex':
  362. return '0x' + config_value[2:].upper()
  363. elif key_type == 'list':
  364. return config_value.split()
  365. elif key_type == 'int':
  366. return int(config_value)
  367. elif key_type == 'str':
  368. return config_value.strip('"').replace('\\"', '"').replace('\\\\', '\\')
  369. elif key_type == 'bcd_version':
  370. major = int(config_value[2:4])
  371. minor = int(config_value[4])
  372. revision = int(config_value[5])
  373. return f'{major}.{minor}.{revision}'
  374. return config_value
  375. def _extract_config_h(info_data, config_c):
  376. """Pull some keyboard information from existing config.h files
  377. """
  378. # Pull in data from the json map
  379. dotty_info = dotty(info_data)
  380. info_config_map = json_load(Path('data/mappings/info_config.hjson'))
  381. for config_key, info_dict in info_config_map.items():
  382. info_key = info_dict['info_key']
  383. key_type = info_dict.get('value_type', 'raw')
  384. try:
  385. replace_with = info_dict.get('replace_with')
  386. if config_key in config_c and info_dict.get('invalid', False):
  387. if replace_with:
  388. _log_error(info_data, '%s in config.h is no longer a valid option and should be replaced with %s' % (config_key, replace_with))
  389. else:
  390. _log_error(info_data, '%s in config.h is no longer a valid option and should be removed' % config_key)
  391. elif config_key in config_c and info_dict.get('deprecated', False):
  392. if replace_with:
  393. _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))
  394. else:
  395. _log_warning(info_data, '%s in config.h is deprecated and will be removed at a later date' % config_key)
  396. if config_key in config_c and info_dict.get('to_json', True):
  397. if dotty_info.get(info_key) and info_dict.get('warn_duplicate', True):
  398. _log_warning(info_data, '%s in config.h is overwriting %s in info.json' % (config_key, info_key))
  399. dotty_info[info_key] = _config_to_json(key_type, config_c[config_key])
  400. except Exception as e:
  401. _log_warning(info_data, f'{config_key}->{info_key}: {e}')
  402. info_data.update(dotty_info)
  403. # Pull data that easily can't be mapped in json
  404. _extract_matrix_info(info_data, config_c)
  405. _extract_audio(info_data, config_c)
  406. _extract_secure_unlock(info_data, config_c)
  407. _extract_split_handedness(info_data, config_c)
  408. _extract_split_transport(info_data, config_c)
  409. _extract_split_right_pins(info_data, config_c)
  410. _extract_encoders(info_data, config_c)
  411. _extract_split_encoders(info_data, config_c)
  412. return info_data
  413. def _process_defaults(info_data):
  414. """Process any additional defaults based on currently discovered information
  415. """
  416. defaults_map = json_load(Path('data/mappings/defaults.hjson'))
  417. for default_type in defaults_map.keys():
  418. thing_map = defaults_map[default_type]
  419. if default_type in info_data:
  420. merged_count = 0
  421. thing_items = thing_map.get(info_data[default_type], {}).items()
  422. for key, value in thing_items:
  423. if key not in info_data:
  424. info_data[key] = value
  425. merged_count += 1
  426. if merged_count == 0 and len(thing_items) > 0:
  427. _log_warning(info_data, 'All defaults for \'%s\' were skipped, potential redundant config or misconfiguration detected' % (default_type))
  428. return info_data
  429. def _extract_rules_mk(info_data, rules):
  430. """Pull some keyboard information from existing rules.mk files
  431. """
  432. info_data['processor'] = rules.get('MCU', info_data.get('processor', 'atmega32u4'))
  433. if info_data['processor'] in CHIBIOS_PROCESSORS:
  434. arm_processor_rules(info_data, rules)
  435. elif info_data['processor'] in LUFA_PROCESSORS + VUSB_PROCESSORS:
  436. avr_processor_rules(info_data, rules)
  437. else:
  438. cli.log.warning("%s: Unknown MCU: %s" % (info_data['keyboard_folder'], info_data['processor']))
  439. unknown_processor_rules(info_data, rules)
  440. # Pull in data from the json map
  441. dotty_info = dotty(info_data)
  442. info_rules_map = json_load(Path('data/mappings/info_rules.hjson'))
  443. for rules_key, info_dict in info_rules_map.items():
  444. info_key = info_dict['info_key']
  445. key_type = info_dict.get('value_type', 'raw')
  446. try:
  447. replace_with = info_dict.get('replace_with')
  448. if rules_key in rules and info_dict.get('invalid', False):
  449. if replace_with:
  450. _log_error(info_data, '%s in rules.mk is no longer a valid option and should be replaced with %s' % (rules_key, replace_with))
  451. else:
  452. _log_error(info_data, '%s in rules.mk is no longer a valid option and should be removed' % rules_key)
  453. elif rules_key in rules and info_dict.get('deprecated', False):
  454. if replace_with:
  455. _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))
  456. else:
  457. _log_warning(info_data, '%s in rules.mk is deprecated and will be removed at a later date' % rules_key)
  458. if rules_key in rules and info_dict.get('to_json', True):
  459. if dotty_info.get(info_key) and info_dict.get('warn_duplicate', True):
  460. _log_warning(info_data, '%s in rules.mk is overwriting %s in info.json' % (rules_key, info_key))
  461. dotty_info[info_key] = _config_to_json(key_type, rules[rules_key])
  462. except Exception as e:
  463. _log_warning(info_data, f'{rules_key}->{info_key}: {e}')
  464. info_data.update(dotty_info)
  465. # Merge in config values that can't be easily mapped
  466. _extract_features(info_data, rules)
  467. return info_data
  468. def find_keyboard_c(keyboard):
  469. """Find all <keyboard>.c files
  470. """
  471. keyboard = Path(keyboard)
  472. current_path = Path('keyboards/')
  473. files = []
  474. for directory in keyboard.parts:
  475. current_path = current_path / directory
  476. keyboard_c_path = current_path / f'{directory}.c'
  477. if keyboard_c_path.exists():
  478. files.append(keyboard_c_path)
  479. return files
  480. def _extract_led_config(info_data, keyboard):
  481. """Scan all <keyboard>.c files for led config
  482. """
  483. cols = info_data['matrix_size']['cols']
  484. rows = info_data['matrix_size']['rows']
  485. # Determine what feature owns g_led_config
  486. features = info_data.get("features", {})
  487. feature = None
  488. if features.get("rgb_matrix", False):
  489. feature = "rgb_matrix"
  490. elif features.get("led_matrix", False):
  491. feature = "led_matrix"
  492. if feature:
  493. # Process
  494. for file in find_keyboard_c(keyboard):
  495. try:
  496. ret = find_led_config(file, cols, rows)
  497. if ret:
  498. info_data[feature] = info_data.get(feature, {})
  499. info_data[feature]["layout"] = ret
  500. except Exception as e:
  501. _log_warning(info_data, f'led_config: {file.name}: {e}')
  502. if info_data[feature].get("layout", None) and not info_data[feature].get("led_count", None):
  503. info_data[feature]["led_count"] = len(info_data[feature]["layout"])
  504. return info_data
  505. def _matrix_size(info_data):
  506. """Add info_data['matrix_size'] if it doesn't exist.
  507. """
  508. if 'matrix_size' not in info_data and 'matrix_pins' in info_data:
  509. info_data['matrix_size'] = {}
  510. if 'direct' in info_data['matrix_pins']:
  511. info_data['matrix_size']['cols'] = len(info_data['matrix_pins']['direct'][0])
  512. info_data['matrix_size']['rows'] = len(info_data['matrix_pins']['direct'])
  513. elif 'cols' in info_data['matrix_pins'] and 'rows' in info_data['matrix_pins']:
  514. info_data['matrix_size']['cols'] = len(info_data['matrix_pins']['cols'])
  515. info_data['matrix_size']['rows'] = len(info_data['matrix_pins']['rows'])
  516. # Assumption of split common
  517. if 'split' in info_data:
  518. if info_data['split'].get('enabled', False):
  519. info_data['matrix_size']['rows'] *= 2
  520. return info_data
  521. def _check_matrix(info_data):
  522. """Check the matrix to ensure that row/column count is consistent.
  523. """
  524. if 'matrix_pins' in info_data and 'matrix_size' in info_data:
  525. actual_col_count = info_data['matrix_size'].get('cols', 0)
  526. actual_row_count = info_data['matrix_size'].get('rows', 0)
  527. col_count = row_count = 0
  528. if 'direct' in info_data['matrix_pins']:
  529. col_count = len(info_data['matrix_pins']['direct'][0])
  530. row_count = len(info_data['matrix_pins']['direct'])
  531. elif 'cols' in info_data['matrix_pins'] and 'rows' in info_data['matrix_pins']:
  532. col_count = len(info_data['matrix_pins']['cols'])
  533. row_count = len(info_data['matrix_pins']['rows'])
  534. if col_count != actual_col_count and col_count != (actual_col_count / 2):
  535. # FIXME: once we can we should detect if split is enabled to do the actual_col_count/2 check.
  536. _log_error(info_data, f'MATRIX_COLS is inconsistent with the size of MATRIX_COL_PINS: {col_count} != {actual_col_count}')
  537. if row_count != actual_row_count and row_count != (actual_row_count / 2):
  538. # FIXME: once we can we should detect if split is enabled to do the actual_row_count/2 check.
  539. _log_error(info_data, f'MATRIX_ROWS is inconsistent with the size of MATRIX_ROW_PINS: {row_count} != {actual_row_count}')
  540. def _search_keyboard_h(keyboard):
  541. keyboard = Path(keyboard)
  542. current_path = Path('keyboards/')
  543. aliases = {}
  544. layouts = {}
  545. for directory in keyboard.parts:
  546. current_path = current_path / directory
  547. keyboard_h = '%s.h' % (directory,)
  548. keyboard_h_path = current_path / keyboard_h
  549. if keyboard_h_path.exists():
  550. new_layouts, new_aliases = find_layouts(keyboard_h_path)
  551. layouts.update(new_layouts)
  552. for alias, alias_text in new_aliases.items():
  553. if alias_text in layouts:
  554. aliases[alias] = alias_text
  555. return layouts, aliases
  556. def _log_error(info_data, message):
  557. """Send an error message to both JSON and the log.
  558. """
  559. info_data['parse_errors'].append(message)
  560. cli.log.error('%s: %s', info_data.get('keyboard_folder', 'Unknown Keyboard!'), message)
  561. def _log_warning(info_data, message):
  562. """Send a warning message to both JSON and the log.
  563. """
  564. info_data['parse_warnings'].append(message)
  565. cli.log.warning('%s: %s', info_data.get('keyboard_folder', 'Unknown Keyboard!'), message)
  566. def arm_processor_rules(info_data, rules):
  567. """Setup the default info for an ARM board.
  568. """
  569. info_data['processor_type'] = 'arm'
  570. info_data['protocol'] = 'ChibiOS'
  571. info_data['platform_key'] = 'chibios'
  572. if 'STM32' in info_data['processor']:
  573. info_data['platform'] = 'STM32'
  574. elif 'MCU_SERIES' in rules:
  575. info_data['platform'] = rules['MCU_SERIES']
  576. elif 'ARM_ATSAM' in rules:
  577. info_data['platform'] = 'ARM_ATSAM'
  578. info_data['platform_key'] = 'arm_atsam'
  579. return info_data
  580. def avr_processor_rules(info_data, rules):
  581. """Setup the default info for an AVR board.
  582. """
  583. info_data['processor_type'] = 'avr'
  584. info_data['platform'] = rules['ARCH'] if 'ARCH' in rules else 'unknown'
  585. info_data['platform_key'] = 'avr'
  586. info_data['protocol'] = 'V-USB' if info_data['processor'] in VUSB_PROCESSORS else 'LUFA'
  587. # FIXME(fauxpark/anyone): Eventually we should detect the protocol by looking at PROTOCOL inherited from mcu_selection.mk:
  588. # info_data['protocol'] = 'V-USB' if rules.get('PROTOCOL') == 'VUSB' else 'LUFA'
  589. return info_data
  590. def unknown_processor_rules(info_data, rules):
  591. """Setup the default keyboard info for unknown boards.
  592. """
  593. info_data['bootloader'] = 'unknown'
  594. info_data['platform'] = 'unknown'
  595. info_data['processor'] = 'unknown'
  596. info_data['processor_type'] = 'unknown'
  597. info_data['protocol'] = 'unknown'
  598. return info_data
  599. def merge_info_jsons(keyboard, info_data):
  600. """Return a merged copy of all the info.json files for a keyboard.
  601. """
  602. for info_file in find_info_json(keyboard):
  603. # Load and validate the JSON data
  604. new_info_data = json_load(info_file)
  605. if not isinstance(new_info_data, dict):
  606. _log_error(info_data, "Invalid file %s, root object should be a dictionary." % (str(info_file),))
  607. continue
  608. try:
  609. validate(new_info_data, 'qmk.keyboard.v1')
  610. except jsonschema.ValidationError as e:
  611. json_path = '.'.join([str(p) for p in e.absolute_path])
  612. cli.log.error('Not including data from file: %s', info_file)
  613. cli.log.error('\t%s: %s', json_path, e.message)
  614. continue
  615. # Merge layout data in
  616. if 'layout_aliases' in new_info_data:
  617. info_data['layout_aliases'] = {**info_data.get('layout_aliases', {}), **new_info_data['layout_aliases']}
  618. del new_info_data['layout_aliases']
  619. for layout_name, layout in new_info_data.get('layouts', {}).items():
  620. if layout_name in info_data.get('layout_aliases', {}):
  621. _log_warning(info_data, f"info.json uses alias name {layout_name} instead of {info_data['layout_aliases'][layout_name]}")
  622. layout_name = info_data['layout_aliases'][layout_name]
  623. if layout_name in info_data['layouts']:
  624. if len(info_data['layouts'][layout_name]['layout']) != len(layout['layout']):
  625. msg = 'Number of keys for %s does not match! info.json specifies %d keys, C macro specifies %d'
  626. _log_error(info_data, msg % (layout_name, len(layout['layout']), len(info_data['layouts'][layout_name]['layout'])))
  627. else:
  628. info_data['layouts'][layout_name]['json_layout'] = True
  629. for new_key, existing_key in zip(layout['layout'], info_data['layouts'][layout_name]['layout']):
  630. existing_key.update(new_key)
  631. else:
  632. if not all('matrix' in key_data.keys() for key_data in layout['layout']):
  633. _log_error(info_data, f'Layout "{layout_name}" has no "matrix" definition in either "info.json" or "<keyboard>.h"!')
  634. else:
  635. layout['c_macro'] = False
  636. layout['json_layout'] = True
  637. info_data['layouts'][layout_name] = layout
  638. # Update info_data with the new data
  639. if 'layouts' in new_info_data:
  640. del new_info_data['layouts']
  641. deep_update(info_data, new_info_data)
  642. return info_data
  643. def find_info_json(keyboard):
  644. """Finds all the info.json files associated with a keyboard.
  645. """
  646. # Find the most specific first
  647. base_path = Path('keyboards')
  648. keyboard_path = base_path / keyboard
  649. keyboard_parent = keyboard_path.parent
  650. info_jsons = [keyboard_path / 'info.json']
  651. # Add DEFAULT_FOLDER before parents, if present
  652. rules = rules_mk(keyboard)
  653. if 'DEFAULT_FOLDER' in rules:
  654. info_jsons.append(Path(rules['DEFAULT_FOLDER']) / 'info.json')
  655. # Add in parent folders for least specific
  656. for _ in range(5):
  657. if keyboard_parent == base_path:
  658. break
  659. info_jsons.append(keyboard_parent / 'info.json')
  660. keyboard_parent = keyboard_parent.parent
  661. # Return a list of the info.json files that actually exist
  662. return [info_json for info_json in info_jsons if info_json.exists()]
  663. def keymap_json_config(keyboard, keymap):
  664. """Extract keymap level config
  665. """
  666. # TODO: resolve keymap.py and info.py circular dependencies
  667. from qmk.keymap import locate_keymap
  668. keymap_folder = locate_keymap(keyboard, keymap).parent
  669. km_info_json = parse_configurator_json(keymap_folder / 'keymap.json')
  670. return km_info_json.get('config', {})
  671. def keymap_json(keyboard, keymap):
  672. """Generate the info.json data for a specific keymap.
  673. """
  674. # TODO: resolve keymap.py and info.py circular dependencies
  675. from qmk.keymap import locate_keymap
  676. keymap_folder = locate_keymap(keyboard, keymap).parent
  677. # Files to scan
  678. keymap_config = keymap_folder / 'config.h'
  679. keymap_rules = keymap_folder / 'rules.mk'
  680. keymap_file = keymap_folder / 'keymap.json'
  681. # Build the info.json file
  682. kb_info_json = info_json(keyboard)
  683. # Merge in the data from keymap.json
  684. km_info_json = keymap_json_config(keyboard, keymap) if keymap_file.exists() else {}
  685. deep_update(kb_info_json, km_info_json)
  686. # Merge in the data from config.h, and rules.mk
  687. _extract_rules_mk(kb_info_json, parse_rules_mk_file(keymap_rules))
  688. _extract_config_h(kb_info_json, parse_config_h_file(keymap_config))
  689. return kb_info_json