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.

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