logo

qmk_firmware

custom branch of QMK firmware git clone https://anongit.hacktivis.me/git/qmk_firmware.git

build_targets.py (11832B)


  1. # Copyright 2023-2024 Nick Brassel (@tzarc)
  2. # SPDX-License-Identifier: GPL-2.0-or-later
  3. import json
  4. import shutil
  5. from typing import Dict, List, Union
  6. from pathlib import Path
  7. from dotty_dict import dotty, Dotty
  8. from milc import cli
  9. from qmk.constants import QMK_FIRMWARE, INTERMEDIATE_OUTPUT_PREFIX, HAS_QMK_USERSPACE, QMK_USERSPACE
  10. from qmk.commands import find_make, get_make_parallel_args, parse_configurator_json
  11. from qmk.keyboard import keyboard_folder
  12. from qmk.info import keymap_json
  13. from qmk.keymap import locate_keymap
  14. from qmk.path import is_under_qmk_firmware, is_under_qmk_userspace, unix_style_path
  15. from qmk.compilation_database import write_compilation_database
  16. # These must be kept in the order in which they're applied to $(TARGET) in the makefiles in order to ensure consistency.
  17. TARGET_FILENAME_MODIFIERS = ['FORCE_LAYOUT', 'CONVERT_TO']
  18. class BuildTarget:
  19. def __init__(self, keyboard: str, keymap: str, json: Union[dict, Dotty] = None):
  20. self._keyboard = keyboard_folder(keyboard)
  21. self._keyboard_safe = self._keyboard.replace('/', '_')
  22. self._keymap = keymap
  23. self._parallel = 1
  24. self._clean = False
  25. self._compiledb = False
  26. self._extra_args = {}
  27. self._json = json.to_dict() if isinstance(json, Dotty) else json
  28. def __str__(self):
  29. return f'{self.keyboard}:{self.keymap}'
  30. def __repr__(self):
  31. if len(self._extra_args.items()) > 0:
  32. return f'BuildTarget(keyboard={self.keyboard}, keymap={self.keymap}, extra_args={json.dumps(self._extra_args, sort_keys=True)})'
  33. return f'BuildTarget(keyboard={self.keyboard}, keymap={self.keymap})'
  34. def __lt__(self, __value: object) -> bool:
  35. return self.__repr__() < __value.__repr__()
  36. def __eq__(self, __value: object) -> bool:
  37. if not isinstance(__value, BuildTarget):
  38. return False
  39. return self.__repr__() == __value.__repr__()
  40. def __hash__(self) -> int:
  41. return self.__repr__().__hash__()
  42. def configure(self, parallel: int = None, clean: bool = None, compiledb: bool = None) -> None:
  43. if parallel is not None:
  44. self._parallel = parallel
  45. if clean is not None:
  46. self._clean = clean
  47. if compiledb is not None:
  48. self._compiledb = compiledb
  49. @property
  50. def keyboard(self) -> str:
  51. return self._keyboard
  52. @property
  53. def keymap(self) -> str:
  54. return self._keymap
  55. @property
  56. def json(self) -> dict:
  57. if not self._json:
  58. self._load_json()
  59. if not self._json:
  60. return {}
  61. return self._json
  62. @property
  63. def dotty(self) -> Dotty:
  64. return dotty(self.json)
  65. @property
  66. def extra_args(self) -> Dict[str, str]:
  67. return {k: v for k, v in self._extra_args.items()}
  68. @extra_args.setter
  69. def extra_args(self, ex_args: Dict[str, str]):
  70. if ex_args is not None and isinstance(ex_args, dict):
  71. self._extra_args = {k: v for k, v in ex_args.items()}
  72. def target_name(self, **env_vars) -> str:
  73. # Work out the intended target name
  74. target = f'{self._keyboard_safe}_{self.keymap}'
  75. vars = self._all_vars(**env_vars)
  76. for modifier in TARGET_FILENAME_MODIFIERS:
  77. if modifier in vars:
  78. target += f"_{vars[modifier]}"
  79. return target
  80. def _all_vars(self, **env_vars) -> Dict[str, str]:
  81. vars = {k: v for k, v in env_vars.items()}
  82. for k, v in self._extra_args.items():
  83. vars[k] = v
  84. return vars
  85. def _intermediate_output(self, **env_vars) -> Path:
  86. return Path(f'{INTERMEDIATE_OUTPUT_PREFIX}{self.target_name(**env_vars)}')
  87. def _common_make_args(self, dry_run: bool = False, build_target: str = None, **env_vars):
  88. compile_args = [
  89. find_make(),
  90. *get_make_parallel_args(self._parallel),
  91. '-r',
  92. '-R',
  93. '-f',
  94. 'builddefs/build_keyboard.mk',
  95. ]
  96. if not cli.config.general.verbose:
  97. compile_args.append('-s')
  98. verbose = 'true' if cli.config.general.verbose else 'false'
  99. color = 'true' if cli.config.general.color else 'false'
  100. if dry_run:
  101. compile_args.append('-n')
  102. if build_target:
  103. compile_args.append(build_target)
  104. compile_args.extend([
  105. f'KEYBOARD={self.keyboard}',
  106. f'KEYMAP={self.keymap}',
  107. f'KEYBOARD_FILESAFE={self._keyboard_safe}',
  108. f'TARGET={self._keyboard_safe}_{self.keymap}', # don't use self.target_name() here, it's rebuilt on the makefile side
  109. f'VERBOSE={verbose}',
  110. f'COLOR={color}',
  111. 'SILENT=false',
  112. 'QMK_BIN="qmk"',
  113. ])
  114. vars = self._all_vars(**env_vars)
  115. for k, v in vars.items():
  116. compile_args.append(f'{k}={v}')
  117. return compile_args
  118. def prepare_build(self, build_target: str = None, dry_run: bool = False, **env_vars) -> None:
  119. raise NotImplementedError("prepare_build() not implemented in base class")
  120. def compile_command(self, build_target: str = None, dry_run: bool = False, **env_vars) -> List[str]:
  121. raise NotImplementedError("compile_command() not implemented in base class")
  122. def generate_compilation_database(self, build_target: str = None, skip_clean: bool = False, **env_vars) -> None:
  123. self.prepare_build(build_target=build_target, **env_vars)
  124. command = self.compile_command(build_target=build_target, dry_run=True, **env_vars)
  125. output_path = QMK_FIRMWARE / 'compile_commands.json'
  126. ret = write_compilation_database(command=command, output_path=output_path, skip_clean=skip_clean, **env_vars)
  127. if ret and output_path.exists() and HAS_QMK_USERSPACE:
  128. shutil.copy(str(output_path), str(QMK_USERSPACE / 'compile_commands.json'))
  129. return ret
  130. def compile(self, build_target: str = None, dry_run: bool = False, **env_vars) -> None:
  131. if self._clean or self._compiledb:
  132. command = [find_make(), "clean"]
  133. if dry_run:
  134. command.append('-n')
  135. cli.log.info('Cleaning with {fg_cyan}%s', ' '.join(command))
  136. cli.run(command, capture_output=False)
  137. if self._compiledb and not dry_run:
  138. self.generate_compilation_database(build_target=build_target, skip_clean=True, **env_vars)
  139. self.prepare_build(build_target=build_target, dry_run=dry_run, **env_vars)
  140. command = self.compile_command(build_target=build_target, **env_vars)
  141. cli.log.info('Compiling keymap with {fg_cyan}%s', ' '.join(command))
  142. if not dry_run:
  143. cli.echo('\n')
  144. ret = cli.run(command, capture_output=False)
  145. if ret.returncode:
  146. return ret.returncode
  147. class KeyboardKeymapBuildTarget(BuildTarget):
  148. def __init__(self, keyboard: str, keymap: str, json: dict = None):
  149. super().__init__(keyboard=keyboard, keymap=keymap, json=json)
  150. def __repr__(self):
  151. if len(self._extra_args.items()) > 0:
  152. return f'KeyboardKeymapTarget(keyboard={self.keyboard}, keymap={self.keymap}, extra_args={self._extra_args})'
  153. return f'KeyboardKeymapTarget(keyboard={self.keyboard}, keymap={self.keymap})'
  154. def _load_json(self):
  155. self._json = keymap_json(self.keyboard, self.keymap)
  156. def prepare_build(self, build_target: str = None, dry_run: bool = False, **env_vars) -> None:
  157. pass
  158. def compile_command(self, build_target: str = None, dry_run: bool = False, **env_vars) -> List[str]:
  159. compile_args = self._common_make_args(dry_run=dry_run, build_target=build_target, **env_vars)
  160. # Need to override the keymap path if the keymap is a userspace directory.
  161. # This also ensures keyboard aliases as per `keyboard_aliases.hjson` still work if the userspace has the keymap
  162. # in an equivalent historical location.
  163. vars = self._all_vars(**env_vars)
  164. keymap_location = locate_keymap(self.keyboard, self.keymap, force_layout=vars.get('FORCE_LAYOUT'))
  165. if is_under_qmk_userspace(keymap_location) and not is_under_qmk_firmware(keymap_location):
  166. keymap_directory = keymap_location.parent
  167. compile_args.extend([
  168. f'MAIN_KEYMAP_PATH_1={unix_style_path(keymap_directory)}',
  169. f'MAIN_KEYMAP_PATH_2={unix_style_path(keymap_directory)}',
  170. f'MAIN_KEYMAP_PATH_3={unix_style_path(keymap_directory)}',
  171. f'MAIN_KEYMAP_PATH_4={unix_style_path(keymap_directory)}',
  172. f'MAIN_KEYMAP_PATH_5={unix_style_path(keymap_directory)}',
  173. ])
  174. return compile_args
  175. class JsonKeymapBuildTarget(BuildTarget):
  176. def __init__(self, json_path):
  177. if isinstance(json_path, Path):
  178. self.json_path = json_path
  179. else:
  180. self.json_path = None
  181. json = parse_configurator_json(json_path) # Will load from stdin if provided
  182. # In case the user passes a keymap.json from a keymap directory directly to the CLI.
  183. # e.g.: qmk compile - < keyboards/clueboard/california/keymaps/default/keymap.json
  184. json["keymap"] = json.get("keymap", "default_json")
  185. super().__init__(keyboard=json['keyboard'], keymap=json['keymap'], json=json)
  186. def __repr__(self):
  187. if len(self._extra_args.items()) > 0:
  188. return f'JsonKeymapTarget(keyboard={self.keyboard}, keymap={self.keymap}, path={self.json_path}, extra_args={self._extra_args})'
  189. return f'JsonKeymapTarget(keyboard={self.keyboard}, keymap={self.keymap}, path={self.json_path})'
  190. def _load_json(self):
  191. pass # Already loaded in constructor
  192. def prepare_build(self, build_target: str = None, dry_run: bool = False, **env_vars) -> None:
  193. intermediate_output = self._intermediate_output(**env_vars)
  194. generated_files_path = intermediate_output / 'src'
  195. keymap_json = generated_files_path / 'keymap.json'
  196. if self._clean:
  197. if intermediate_output.exists():
  198. shutil.rmtree(intermediate_output)
  199. # begin with making the deepest folder in the tree
  200. generated_files_path.mkdir(exist_ok=True, parents=True)
  201. # Compare minified to ensure consistent comparison
  202. new_content = json.dumps(self.json, separators=(',', ':'))
  203. if keymap_json.exists():
  204. old_content = json.dumps(json.loads(keymap_json.read_text(encoding='utf-8')), separators=(',', ':'))
  205. if old_content == new_content:
  206. new_content = None
  207. # Write the keymap.json file if different so timestamps are only updated
  208. # if the content changes -- running `make` won't treat it as modified.
  209. if new_content:
  210. keymap_json.write_text(new_content, encoding='utf-8')
  211. def compile_command(self, build_target: str = None, dry_run: bool = False, **env_vars) -> List[str]:
  212. compile_args = self._common_make_args(dry_run=dry_run, build_target=build_target, **env_vars)
  213. intermediate_output = self._intermediate_output(**env_vars)
  214. generated_files_path = intermediate_output / 'src'
  215. keymap_json = generated_files_path / 'keymap.json'
  216. compile_args.extend([
  217. f'MAIN_KEYMAP_PATH_1={unix_style_path(intermediate_output)}',
  218. f'MAIN_KEYMAP_PATH_2={unix_style_path(intermediate_output)}',
  219. f'MAIN_KEYMAP_PATH_3={unix_style_path(intermediate_output)}',
  220. f'MAIN_KEYMAP_PATH_4={unix_style_path(intermediate_output)}',
  221. f'MAIN_KEYMAP_PATH_5={unix_style_path(intermediate_output)}',
  222. f'KEYMAP_JSON={keymap_json}',
  223. f'KEYMAP_PATH={generated_files_path}',
  224. ])
  225. return compile_args