info.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551
  1. """Functions that help us generate and use info.json files.
  2. """
  3. from glob import glob
  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
  10. from qmk.json_schema import deep_update, json_load, keyboard_validate, keyboard_api_validate
  11. from qmk.keyboard import config_h, rules_mk
  12. from qmk.keymap import list_keymaps
  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 _valid_community_layout(layout):
  18. """Validate that a declared community list exists
  19. """
  20. return (Path('layouts/default') / layout).exists()
  21. def info_json(keyboard):
  22. """Generate the info.json data for a specific keyboard.
  23. """
  24. cur_dir = Path('keyboards')
  25. rules = parse_rules_mk_file(cur_dir / keyboard / 'rules.mk')
  26. if 'DEFAULT_FOLDER' in rules:
  27. keyboard = rules['DEFAULT_FOLDER']
  28. rules = parse_rules_mk_file(cur_dir / keyboard / 'rules.mk', rules)
  29. info_data = {
  30. 'keyboard_name': str(keyboard),
  31. 'keyboard_folder': str(keyboard),
  32. 'keymaps': {},
  33. 'layouts': {},
  34. 'parse_errors': [],
  35. 'parse_warnings': [],
  36. 'maintainer': 'qmk',
  37. }
  38. # Populate the list of JSON keymaps
  39. for keymap in list_keymaps(keyboard, c=False, fullpath=True):
  40. info_data['keymaps'][keymap.name] = {'url': f'https://raw.githubusercontent.com/qmk/qmk_firmware/master/{keymap}/keymap.json'}
  41. # Populate layout data
  42. layouts, aliases = _find_all_layouts(info_data, keyboard)
  43. if aliases:
  44. info_data['layout_aliases'] = aliases
  45. for layout_name, layout_json in layouts.items():
  46. if not layout_name.startswith('LAYOUT_kc'):
  47. layout_json['c_macro'] = True
  48. info_data['layouts'][layout_name] = layout_json
  49. # Merge in the data from info.json, config.h, and rules.mk
  50. info_data = merge_info_jsons(keyboard, info_data)
  51. info_data = _extract_config_h(info_data)
  52. info_data = _extract_rules_mk(info_data)
  53. # Validate against the jsonschema
  54. try:
  55. keyboard_api_validate(info_data)
  56. except jsonschema.ValidationError as e:
  57. json_path = '.'.join([str(p) for p in e.absolute_path])
  58. cli.log.error('Invalid API data: %s: %s: %s', keyboard, json_path, e.message)
  59. exit()
  60. # Make sure we have at least one layout
  61. if not info_data.get('layouts'):
  62. _log_error(info_data, 'No LAYOUTs defined! Need at least one layout defined in the keyboard.h or info.json.')
  63. # Filter out any non-existing community layouts
  64. for layout in info_data.get('community_layouts', []):
  65. if not _valid_community_layout(layout):
  66. # Ignore layout from future checks
  67. info_data['community_layouts'].remove(layout)
  68. _log_error(info_data, 'Claims to support a community layout that does not exist: %s' % (layout))
  69. # Make sure we supply layout macros for the community layouts we claim to support
  70. for layout in info_data.get('community_layouts', []):
  71. layout_name = 'LAYOUT_' + layout
  72. if layout_name not in info_data.get('layouts', {}) and layout_name not in info_data.get('layout_aliases', {}):
  73. _log_error(info_data, 'Claims to support community layout %s but no %s() macro found' % (layout, layout_name))
  74. return info_data
  75. def _extract_features(info_data, rules):
  76. """Find all the features enabled in rules.mk.
  77. """
  78. # Special handling for bootmagic which also supports a "lite" mode.
  79. if rules.get('BOOTMAGIC_ENABLE') == 'lite':
  80. rules['BOOTMAGIC_LITE_ENABLE'] = 'on'
  81. del rules['BOOTMAGIC_ENABLE']
  82. if rules.get('BOOTMAGIC_ENABLE') == 'full':
  83. rules['BOOTMAGIC_ENABLE'] = 'on'
  84. # Skip non-boolean features we haven't implemented special handling for
  85. for feature in 'HAPTIC_ENABLE', 'QWIIC_ENABLE':
  86. if rules.get(feature):
  87. del rules[feature]
  88. # Process the rest of the rules as booleans
  89. for key, value in rules.items():
  90. if key.endswith('_ENABLE'):
  91. key = '_'.join(key.split('_')[:-1]).lower()
  92. value = True if value.lower() in true_values else False if value.lower() in false_values else value
  93. if 'config_h_features' not in info_data:
  94. info_data['config_h_features'] = {}
  95. if 'features' not in info_data:
  96. info_data['features'] = {}
  97. if key in info_data['features']:
  98. _log_warning(info_data, 'Feature %s is specified in both info.json and rules.mk, the rules.mk value wins.' % (key,))
  99. info_data['features'][key] = value
  100. info_data['config_h_features'][key] = value
  101. return info_data
  102. def _pin_name(pin):
  103. """Returns the proper representation for a pin.
  104. """
  105. pin = pin.strip()
  106. if not pin:
  107. return None
  108. elif pin.isdigit():
  109. return int(pin)
  110. elif pin == 'NO_PIN':
  111. return None
  112. elif pin[0] in 'ABCDEFGHIJK' and pin[1].isdigit():
  113. return pin
  114. raise ValueError(f'Invalid pin: {pin}')
  115. def _extract_pins(pins):
  116. """Returns a list of pins from a comma separated string of pins.
  117. """
  118. return [_pin_name(pin) for pin in pins.split(',')]
  119. def _extract_direct_matrix(info_data, direct_pins):
  120. """
  121. """
  122. info_data['matrix_pins'] = {}
  123. direct_pin_array = []
  124. while direct_pins[-1] != '}':
  125. direct_pins = direct_pins[:-1]
  126. for row in direct_pins.split('},{'):
  127. if row.startswith('{'):
  128. row = row[1:]
  129. if row.endswith('}'):
  130. row = row[:-1]
  131. direct_pin_array.append([])
  132. for pin in row.split(','):
  133. if pin == 'NO_PIN':
  134. pin = None
  135. direct_pin_array[-1].append(pin)
  136. return direct_pin_array
  137. def _extract_matrix_info(info_data, config_c):
  138. """Populate the matrix information.
  139. """
  140. row_pins = config_c.get('MATRIX_ROW_PINS', '').replace('{', '').replace('}', '').strip()
  141. col_pins = config_c.get('MATRIX_COL_PINS', '').replace('{', '').replace('}', '').strip()
  142. direct_pins = config_c.get('DIRECT_PINS', '').replace(' ', '')[1:-1]
  143. if 'MATRIX_ROWS' in config_c and 'MATRIX_COLS' in config_c:
  144. if 'matrix_size' in info_data:
  145. _log_warning(info_data, 'Matrix size is specified in both info.json and config.h, the config.h values win.')
  146. info_data['matrix_size'] = {
  147. 'cols': compute(config_c.get('MATRIX_COLS', '0')),
  148. 'rows': compute(config_c.get('MATRIX_ROWS', '0')),
  149. }
  150. if row_pins and col_pins:
  151. if 'matrix_pins' in info_data:
  152. _log_warning(info_data, 'Matrix pins are specified in both info.json and config.h, the config.h values win.')
  153. info_data['matrix_pins'] = {
  154. 'cols': _extract_pins(col_pins),
  155. 'rows': _extract_pins(row_pins),
  156. }
  157. if direct_pins:
  158. if 'matrix_pins' in info_data:
  159. _log_warning(info_data, 'Direct pins are specified in both info.json and config.h, the config.h values win.')
  160. info_data['matrix_pins']['direct'] = _extract_direct_matrix(info_data, direct_pins)
  161. return info_data
  162. def _extract_config_h(info_data):
  163. """Pull some keyboard information from existing config.h files
  164. """
  165. config_c = config_h(info_data['keyboard_folder'])
  166. # Pull in data from the json map
  167. dotty_info = dotty(info_data)
  168. info_config_map = json_load(Path('data/mappings/info_config.json'))
  169. for config_key, info_dict in info_config_map.items():
  170. info_key = info_dict['info_key']
  171. key_type = info_dict.get('value_type', 'str')
  172. try:
  173. if config_key in config_c and info_dict.get('to_json', True):
  174. if dotty_info.get(info_key) and info_dict.get('warn_duplicate', True):
  175. _log_warning(info_data, '%s in config.h is overwriting %s in info.json' % (config_key, info_key))
  176. if key_type.startswith('array'):
  177. if '.' in key_type:
  178. key_type, array_type = key_type.split('.', 1)
  179. else:
  180. array_type = None
  181. config_value = config_c[config_key].replace('{', '').replace('}', '').strip()
  182. if array_type == 'int':
  183. dotty_info[info_key] = list(map(int, config_value.split(',')))
  184. else:
  185. dotty_info[info_key] = config_value.split(',')
  186. elif key_type == 'bool':
  187. dotty_info[info_key] = config_c[config_key] in true_values
  188. elif key_type == 'hex':
  189. dotty_info[info_key] = '0x' + config_c[config_key][2:].upper()
  190. elif key_type == 'list':
  191. dotty_info[info_key] = config_c[config_key].split()
  192. elif key_type == 'int':
  193. dotty_info[info_key] = int(config_c[config_key])
  194. else:
  195. dotty_info[info_key] = config_c[config_key]
  196. except Exception as e:
  197. _log_warning(info_data, f'{config_key}->{info_key}: {e}')
  198. info_data.update(dotty_info)
  199. # Pull data that easily can't be mapped in json
  200. _extract_matrix_info(info_data, config_c)
  201. return info_data
  202. def _extract_rules_mk(info_data):
  203. """Pull some keyboard information from existing rules.mk files
  204. """
  205. rules = rules_mk(info_data['keyboard_folder'])
  206. info_data['processor'] = rules.get('MCU', info_data.get('processor', 'atmega32u4'))
  207. if info_data['processor'] in CHIBIOS_PROCESSORS:
  208. arm_processor_rules(info_data, rules)
  209. elif info_data['processor'] in LUFA_PROCESSORS + VUSB_PROCESSORS:
  210. avr_processor_rules(info_data, rules)
  211. else:
  212. cli.log.warning("%s: Unknown MCU: %s" % (info_data['keyboard_folder'], info_data['processor']))
  213. unknown_processor_rules(info_data, rules)
  214. # Pull in data from the json map
  215. dotty_info = dotty(info_data)
  216. info_rules_map = json_load(Path('data/mappings/info_rules.json'))
  217. for rules_key, info_dict in info_rules_map.items():
  218. info_key = info_dict['info_key']
  219. key_type = info_dict.get('value_type', 'str')
  220. try:
  221. if rules_key in rules and info_dict.get('to_json', True):
  222. if dotty_info.get(info_key) and info_dict.get('warn_duplicate', True):
  223. _log_warning(info_data, '%s in rules.mk is overwriting %s in info.json' % (rules_key, info_key))
  224. if key_type.startswith('array'):
  225. if '.' in key_type:
  226. key_type, array_type = key_type.split('.', 1)
  227. else:
  228. array_type = None
  229. rules_value = rules[rules_key].replace('{', '').replace('}', '').strip()
  230. if array_type == 'int':
  231. dotty_info[info_key] = list(map(int, rules_value.split(',')))
  232. else:
  233. dotty_info[info_key] = rules_value.split(',')
  234. elif key_type == 'list':
  235. dotty_info[info_key] = rules[rules_key].split()
  236. elif key_type == 'bool':
  237. dotty_info[info_key] = rules[rules_key] in true_values
  238. elif key_type == 'hex':
  239. dotty_info[info_key] = '0x' + rules[rules_key][2:].upper()
  240. elif key_type == 'int':
  241. dotty_info[info_key] = int(rules[rules_key])
  242. else:
  243. dotty_info[info_key] = rules[rules_key]
  244. except Exception as e:
  245. _log_warning(info_data, f'{rules_key}->{info_key}: {e}')
  246. info_data.update(dotty_info)
  247. # Merge in config values that can't be easily mapped
  248. _extract_features(info_data, rules)
  249. return info_data
  250. def _merge_layouts(info_data, new_info_data):
  251. """Merge new_info_data into info_data in an intelligent way.
  252. """
  253. for layout_name, layout_json in new_info_data['layouts'].items():
  254. if layout_name in info_data['layouts']:
  255. # Pull in layouts we have a macro for
  256. if len(info_data['layouts'][layout_name]['layout']) != len(layout_json['layout']):
  257. msg = '%s: %s: Number of elements in info.json does not match! info.json:%s != %s:%s'
  258. _log_error(info_data, msg % (info_data['keyboard_folder'], layout_name, len(layout_json['layout']), layout_name, len(info_data['layouts'][layout_name]['layout'])))
  259. else:
  260. for i, key in enumerate(info_data['layouts'][layout_name]['layout']):
  261. key.update(layout_json['layout'][i])
  262. else:
  263. # Pull in layouts that have matrix data
  264. missing_matrix = False
  265. for key in layout_json.get('layout', {}):
  266. if 'matrix' not in key:
  267. missing_matrix = True
  268. if not missing_matrix:
  269. if layout_name in info_data['layouts']:
  270. # Update an existing layout with new data
  271. for i, key in enumerate(info_data['layouts'][layout_name]['layout']):
  272. key.update(layout_json['layout'][i])
  273. else:
  274. # Copy in the new layout wholesale
  275. layout_json['c_macro'] = False
  276. info_data['layouts'][layout_name] = layout_json
  277. return info_data
  278. def _search_keyboard_h(path):
  279. current_path = Path('keyboards/')
  280. aliases = {}
  281. layouts = {}
  282. for directory in path.parts:
  283. current_path = current_path / directory
  284. keyboard_h = '%s.h' % (directory,)
  285. keyboard_h_path = current_path / keyboard_h
  286. if keyboard_h_path.exists():
  287. new_layouts, new_aliases = find_layouts(keyboard_h_path)
  288. layouts.update(new_layouts)
  289. for alias, alias_text in new_aliases.items():
  290. if alias_text in layouts:
  291. aliases[alias] = alias_text
  292. return layouts, aliases
  293. def _find_all_layouts(info_data, keyboard):
  294. """Looks for layout macros associated with this keyboard.
  295. """
  296. layouts, aliases = _search_keyboard_h(Path(keyboard))
  297. if not layouts:
  298. # If we don't find any layouts from info.json or keyboard.h we widen our search. This is error prone which is why we want to encourage people to follow the standard above.
  299. info_data['parse_warnings'].append('%s: Falling back to searching for KEYMAP/LAYOUT macros.' % (keyboard))
  300. for file in glob('keyboards/%s/*.h' % keyboard):
  301. if file.endswith('.h'):
  302. these_layouts, these_aliases = find_layouts(file)
  303. if these_layouts:
  304. layouts.update(these_layouts)
  305. for alias, alias_text in these_aliases.items():
  306. if alias_text in layouts:
  307. aliases[alias] = alias_text
  308. return layouts, aliases
  309. def _log_error(info_data, message):
  310. """Send an error message to both JSON and the log.
  311. """
  312. info_data['parse_errors'].append(message)
  313. cli.log.error('%s: %s', info_data.get('keyboard_folder', 'Unknown Keyboard!'), message)
  314. def _log_warning(info_data, message):
  315. """Send a warning message to both JSON and the log.
  316. """
  317. info_data['parse_warnings'].append(message)
  318. cli.log.warning('%s: %s', info_data.get('keyboard_folder', 'Unknown Keyboard!'), message)
  319. def arm_processor_rules(info_data, rules):
  320. """Setup the default info for an ARM board.
  321. """
  322. info_data['processor_type'] = 'arm'
  323. info_data['protocol'] = 'ChibiOS'
  324. if 'bootloader' not in info_data:
  325. if 'STM32' in info_data['processor']:
  326. info_data['bootloader'] = 'stm32-dfu'
  327. else:
  328. info_data['bootloader'] = 'unknown'
  329. if 'STM32' in info_data['processor']:
  330. info_data['platform'] = 'STM32'
  331. elif 'MCU_SERIES' in rules:
  332. info_data['platform'] = rules['MCU_SERIES']
  333. elif 'ARM_ATSAM' in rules:
  334. info_data['platform'] = 'ARM_ATSAM'
  335. return info_data
  336. def avr_processor_rules(info_data, rules):
  337. """Setup the default info for an AVR board.
  338. """
  339. info_data['processor_type'] = 'avr'
  340. info_data['platform'] = rules['ARCH'] if 'ARCH' in rules else 'unknown'
  341. info_data['protocol'] = 'V-USB' if rules.get('MCU') in VUSB_PROCESSORS else 'LUFA'
  342. if 'bootloader' not in info_data:
  343. info_data['bootloader'] = 'atmel-dfu'
  344. # FIXME(fauxpark/anyone): Eventually we should detect the protocol by looking at PROTOCOL inherited from mcu_selection.mk:
  345. # info_data['protocol'] = 'V-USB' if rules.get('PROTOCOL') == 'VUSB' else 'LUFA'
  346. return info_data
  347. def unknown_processor_rules(info_data, rules):
  348. """Setup the default keyboard info for unknown boards.
  349. """
  350. info_data['bootloader'] = 'unknown'
  351. info_data['platform'] = 'unknown'
  352. info_data['processor'] = 'unknown'
  353. info_data['processor_type'] = 'unknown'
  354. info_data['protocol'] = 'unknown'
  355. return info_data
  356. def merge_info_jsons(keyboard, info_data):
  357. """Return a merged copy of all the info.json files for a keyboard.
  358. """
  359. for info_file in find_info_json(keyboard):
  360. # Load and validate the JSON data
  361. new_info_data = json_load(info_file)
  362. if not isinstance(new_info_data, dict):
  363. _log_error(info_data, "Invalid file %s, root object should be a dictionary." % (str(info_file),))
  364. continue
  365. try:
  366. keyboard_validate(new_info_data)
  367. except jsonschema.ValidationError as e:
  368. json_path = '.'.join([str(p) for p in e.absolute_path])
  369. cli.log.error('Not including data from file: %s', info_file)
  370. cli.log.error('\t%s: %s', json_path, e.message)
  371. continue
  372. # Merge layout data in
  373. if 'layout_aliases' in new_info_data:
  374. info_data['layout_aliases'] = {**info_data.get('layout_aliases', {}), **new_info_data['layout_aliases']}
  375. del new_info_data['layout_aliases']
  376. for layout_name, layout in new_info_data.get('layouts', {}).items():
  377. if layout_name in info_data.get('layout_aliases', {}):
  378. _log_warning(info_data, f"info.json uses alias name {layout_name} instead of {info_data['layout_aliases'][layout_name]}")
  379. layout_name = info_data['layout_aliases'][layout_name]
  380. if layout_name in info_data['layouts']:
  381. for new_key, existing_key in zip(layout['layout'], info_data['layouts'][layout_name]['layout']):
  382. existing_key.update(new_key)
  383. else:
  384. layout['c_macro'] = False
  385. info_data['layouts'][layout_name] = layout
  386. # Update info_data with the new data
  387. if 'layouts' in new_info_data:
  388. del new_info_data['layouts']
  389. deep_update(info_data, new_info_data)
  390. return info_data
  391. def find_info_json(keyboard):
  392. """Finds all the info.json files associated with a keyboard.
  393. """
  394. # Find the most specific first
  395. base_path = Path('keyboards')
  396. keyboard_path = base_path / keyboard
  397. keyboard_parent = keyboard_path.parent
  398. info_jsons = [keyboard_path / 'info.json']
  399. # Add DEFAULT_FOLDER before parents, if present
  400. rules = rules_mk(keyboard)
  401. if 'DEFAULT_FOLDER' in rules:
  402. info_jsons.append(Path(rules['DEFAULT_FOLDER']) / 'info.json')
  403. # Add in parent folders for least specific
  404. for _ in range(5):
  405. info_jsons.append(keyboard_parent / 'info.json')
  406. if keyboard_parent.parent == base_path:
  407. break
  408. keyboard_parent = keyboard_parent.parent
  409. # Return a list of the info.json files that actually exist
  410. return [info_json for info_json in info_jsons if info_json.exists()]