config_h.py 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. """Used by the make system to generate info_config.h from info.json.
  2. """
  3. from pathlib import Path
  4. from dotty_dict import dotty
  5. from milc import cli
  6. from qmk.info import info_json, keymap_json_config
  7. from qmk.json_schema import json_load
  8. from qmk.keyboard import keyboard_completer, keyboard_folder
  9. from qmk.commands import dump_lines
  10. from qmk.path import normpath
  11. from qmk.constants import GPL2_HEADER_C_LIKE, GENERATED_HEADER_C_LIKE
  12. def direct_pins(direct_pins, postfix):
  13. """Return the config.h lines that set the direct pins.
  14. """
  15. rows = []
  16. for row in direct_pins:
  17. cols = ','.join(map(str, [col or 'NO_PIN' for col in row]))
  18. rows.append('{' + cols + '}')
  19. return f"""
  20. #ifndef DIRECT_PINS{postfix}
  21. # define DIRECT_PINS{postfix} {{ {", ".join(rows)} }}
  22. #endif // DIRECT_PINS{postfix}
  23. """
  24. def pin_array(define, pins, postfix):
  25. """Return the config.h lines that set a pin array.
  26. """
  27. pin_array = ', '.join(map(str, [pin or 'NO_PIN' for pin in pins]))
  28. return f"""
  29. #ifndef {define}_PINS{postfix}
  30. # define {define}_PINS{postfix} {{ {pin_array} }}
  31. #endif // {define}_PINS{postfix}
  32. """
  33. def matrix_pins(matrix_pins, postfix=''):
  34. """Add the matrix config to the config.h.
  35. """
  36. pins = []
  37. if 'direct' in matrix_pins:
  38. pins.append(direct_pins(matrix_pins['direct'], postfix))
  39. if 'cols' in matrix_pins:
  40. pins.append(pin_array('MATRIX_COL', matrix_pins['cols'], postfix))
  41. if 'rows' in matrix_pins:
  42. pins.append(pin_array('MATRIX_ROW', matrix_pins['rows'], postfix))
  43. return '\n'.join(pins)
  44. def generate_matrix_size(kb_info_json, config_h_lines):
  45. """Add the matrix size to the config.h.
  46. """
  47. if 'matrix_pins' in kb_info_json:
  48. col_count = kb_info_json['matrix_size']['cols']
  49. row_count = kb_info_json['matrix_size']['rows']
  50. config_h_lines.append(f"""
  51. #ifndef MATRIX_COLS
  52. # define MATRIX_COLS {col_count}
  53. #endif // MATRIX_COLS
  54. #ifndef MATRIX_ROWS
  55. # define MATRIX_ROWS {row_count}
  56. #endif // MATRIX_ROWS
  57. """)
  58. def generate_config_items(kb_info_json, config_h_lines):
  59. """Iterate through the info_config map to generate basic config values.
  60. """
  61. info_config_map = json_load(Path('data/mappings/info_config.json'))
  62. for config_key, info_dict in info_config_map.items():
  63. info_key = info_dict['info_key']
  64. key_type = info_dict.get('value_type', 'raw')
  65. to_config = info_dict.get('to_config', True)
  66. if not to_config:
  67. continue
  68. try:
  69. config_value = kb_info_json[info_key]
  70. except KeyError:
  71. continue
  72. if key_type.startswith('array.array'):
  73. config_h_lines.append('')
  74. config_h_lines.append(f'#ifndef {config_key}')
  75. config_h_lines.append(f'# define {config_key} {{ {", ".join(["{" + ",".join(list(map(str, x))) + "}" for x in config_value])} }}')
  76. config_h_lines.append(f'#endif // {config_key}')
  77. elif key_type.startswith('array'):
  78. config_h_lines.append('')
  79. config_h_lines.append(f'#ifndef {config_key}')
  80. config_h_lines.append(f'# define {config_key} {{ {", ".join(map(str, config_value))} }}')
  81. config_h_lines.append(f'#endif // {config_key}')
  82. elif key_type == 'bool':
  83. if config_value:
  84. config_h_lines.append('')
  85. config_h_lines.append(f'#ifndef {config_key}')
  86. config_h_lines.append(f'# define {config_key}')
  87. config_h_lines.append(f'#endif // {config_key}')
  88. elif key_type == 'mapping':
  89. for key, value in config_value.items():
  90. config_h_lines.append('')
  91. config_h_lines.append(f'#ifndef {key}')
  92. config_h_lines.append(f'# define {key} {value}')
  93. config_h_lines.append(f'#endif // {key}')
  94. elif key_type == 'str':
  95. config_h_lines.append('')
  96. config_h_lines.append(f'#ifndef {config_key}')
  97. config_h_lines.append(f'# define {config_key} "{config_value}"')
  98. config_h_lines.append(f'#endif // {config_key}')
  99. elif key_type == 'bcd_version':
  100. (major, minor, revision) = config_value.split('.')
  101. config_h_lines.append('')
  102. config_h_lines.append(f'#ifndef {config_key}')
  103. config_h_lines.append(f'# define {config_key} 0x{major.zfill(2)}{minor}{revision}')
  104. config_h_lines.append(f'#endif // {config_key}')
  105. else:
  106. config_h_lines.append('')
  107. config_h_lines.append(f'#ifndef {config_key}')
  108. config_h_lines.append(f'# define {config_key} {config_value}')
  109. config_h_lines.append(f'#endif // {config_key}')
  110. def generate_encoder_config(encoder_json, config_h_lines, postfix=''):
  111. """Generate the config.h lines for encoders."""
  112. a_pads = []
  113. b_pads = []
  114. resolutions = []
  115. for encoder in encoder_json.get("rotary", []):
  116. a_pads.append(encoder["pin_a"])
  117. b_pads.append(encoder["pin_b"])
  118. resolutions.append(str(encoder.get("resolution", 4)))
  119. config_h_lines.append(f'#ifndef ENCODERS_PAD_A{postfix}')
  120. config_h_lines.append(f'# define ENCODERS_PAD_A{postfix} {{ { ", ".join(a_pads) } }}')
  121. config_h_lines.append(f'#endif // ENCODERS_PAD_A{postfix}')
  122. config_h_lines.append(f'#ifndef ENCODERS_PAD_B{postfix}')
  123. config_h_lines.append(f'# define ENCODERS_PAD_B{postfix} {{ { ", ".join(b_pads) } }}')
  124. config_h_lines.append(f'#endif // ENCODERS_PAD_B{postfix}')
  125. config_h_lines.append(f'#ifndef ENCODER_RESOLUTIONS{postfix}')
  126. config_h_lines.append(f'# define ENCODER_RESOLUTIONS{postfix} {{ { ", ".join(resolutions) } }}')
  127. config_h_lines.append(f'#endif // ENCODER_RESOLUTIONS{postfix}')
  128. def generate_split_config(kb_info_json, config_h_lines):
  129. """Generate the config.h lines for split boards."""
  130. if 'primary' in kb_info_json['split']:
  131. if kb_info_json['split']['primary'] in ('left', 'right'):
  132. config_h_lines.append('')
  133. config_h_lines.append('#ifndef MASTER_LEFT')
  134. config_h_lines.append('# ifndef MASTER_RIGHT')
  135. if kb_info_json['split']['primary'] == 'left':
  136. config_h_lines.append('# define MASTER_LEFT')
  137. elif kb_info_json['split']['primary'] == 'right':
  138. config_h_lines.append('# define MASTER_RIGHT')
  139. config_h_lines.append('# endif // MASTER_RIGHT')
  140. config_h_lines.append('#endif // MASTER_LEFT')
  141. elif kb_info_json['split']['primary'] == 'pin':
  142. config_h_lines.append('')
  143. config_h_lines.append('#ifndef SPLIT_HAND_PIN')
  144. config_h_lines.append('# define SPLIT_HAND_PIN')
  145. config_h_lines.append('#endif // SPLIT_HAND_PIN')
  146. elif kb_info_json['split']['primary'] == 'matrix_grid':
  147. config_h_lines.append('')
  148. config_h_lines.append('#ifndef SPLIT_HAND_MATRIX_GRID')
  149. config_h_lines.append('# define SPLIT_HAND_MATRIX_GRID {%s}' % (','.join(kb_info_json["split"]["matrix_grid"],)))
  150. config_h_lines.append('#endif // SPLIT_HAND_MATRIX_GRID')
  151. elif kb_info_json['split']['primary'] == 'eeprom':
  152. config_h_lines.append('')
  153. config_h_lines.append('#ifndef EE_HANDS')
  154. config_h_lines.append('# define EE_HANDS')
  155. config_h_lines.append('#endif // EE_HANDS')
  156. if 'protocol' in kb_info_json['split'].get('transport', {}):
  157. if kb_info_json['split']['transport']['protocol'] == 'i2c':
  158. config_h_lines.append('')
  159. config_h_lines.append('#ifndef USE_I2C')
  160. config_h_lines.append('# define USE_I2C')
  161. config_h_lines.append('#endif // USE_I2C')
  162. if 'right' in kb_info_json['split'].get('matrix_pins', {}):
  163. config_h_lines.append(matrix_pins(kb_info_json['split']['matrix_pins']['right'], '_RIGHT'))
  164. if 'right' in kb_info_json['split'].get('encoder', {}):
  165. generate_encoder_config(kb_info_json['split']['encoder']['right'], config_h_lines, '_RIGHT')
  166. @cli.argument('-o', '--output', arg_only=True, type=normpath, help='File to write to')
  167. @cli.argument('-q', '--quiet', arg_only=True, action='store_true', help="Quiet mode, only output error messages")
  168. @cli.argument('-kb', '--keyboard', arg_only=True, type=keyboard_folder, completer=keyboard_completer, required=True, help='Keyboard to generate config.h for.')
  169. @cli.argument('-km', '--keymap', arg_only=True, help='Keymap to generate config.h for.')
  170. @cli.subcommand('Used by the make system to generate info_config.h from info.json', hidden=True)
  171. def generate_config_h(cli):
  172. """Generates the info_config.h file.
  173. """
  174. # Determine our keyboard/keymap
  175. if cli.args.keymap:
  176. kb_info_json = dotty(keymap_json_config(cli.args.keyboard, cli.args.keymap))
  177. else:
  178. kb_info_json = dotty(info_json(cli.args.keyboard))
  179. # Build the info_config.h file.
  180. config_h_lines = [GPL2_HEADER_C_LIKE, GENERATED_HEADER_C_LIKE, '#pragma once']
  181. generate_config_items(kb_info_json, config_h_lines)
  182. generate_matrix_size(kb_info_json, config_h_lines)
  183. if 'matrix_pins' in kb_info_json:
  184. config_h_lines.append(matrix_pins(kb_info_json['matrix_pins']))
  185. if 'encoder' in kb_info_json:
  186. generate_encoder_config(kb_info_json['encoder'], config_h_lines)
  187. if 'split' in kb_info_json:
  188. generate_split_config(kb_info_json, config_h_lines)
  189. # Show the results
  190. dump_lines(cli.args.output, config_h_lines, cli.args.quiet)