list_boards.py 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. #!/usr/bin/env python3
  2. # Copyright (c) 2020 Nordic Semiconductor ASA
  3. # SPDX-License-Identifier: Apache-2.0
  4. import argparse
  5. from collections import defaultdict
  6. import itertools
  7. from pathlib import Path
  8. from typing import NamedTuple
  9. ZEPHYR_BASE = Path(__file__).resolve().parents[1]
  10. #
  11. # This is shared code between the build system's 'boards' target
  12. # and the 'west boards' extension command. If you change it, make
  13. # sure to test both ways it can be used.
  14. #
  15. # (It's done this way to keep west optional, making it possible to run
  16. # 'ninja boards' in a build directory without west installed.)
  17. #
  18. class Board(NamedTuple):
  19. name: str
  20. arch: str
  21. dir: Path
  22. def board_key(board):
  23. return board.name
  24. def find_arch2boards(args):
  25. arch2board_set = find_arch2board_set(args)
  26. return {arch: sorted(arch2board_set[arch], key=board_key)
  27. for arch in arch2board_set}
  28. def find_boards(args):
  29. return sorted(itertools.chain(*find_arch2board_set(args).values()),
  30. key=board_key)
  31. def find_arch2board_set(args):
  32. arches = sorted(find_arches(args))
  33. ret = defaultdict(set)
  34. for root in itertools.chain([ZEPHYR_BASE], args.board_roots):
  35. for arch, boards in find_arch2board_set_in(root, arches).items():
  36. ret[arch] |= boards
  37. return ret
  38. def find_arches(args):
  39. arch_set = find_arches_in(ZEPHYR_BASE)
  40. for root in args.arch_roots:
  41. arch_set |= find_arches_in(root)
  42. return arch_set
  43. def find_arches_in(root):
  44. ret = set()
  45. arch = root / 'arch'
  46. common = arch / 'common'
  47. if not arch.is_dir():
  48. return ret
  49. for maybe_arch in arch.iterdir():
  50. if not maybe_arch.is_dir() or maybe_arch == common:
  51. continue
  52. ret.add(maybe_arch.name)
  53. return ret
  54. def find_arch2board_set_in(root, arches):
  55. ret = defaultdict(set)
  56. boards = root / 'boards'
  57. for arch in arches:
  58. if not (boards / arch).is_dir():
  59. continue
  60. for maybe_board in (boards / arch).iterdir():
  61. if not maybe_board.is_dir():
  62. continue
  63. for maybe_defconfig in maybe_board.iterdir():
  64. file_name = maybe_defconfig.name
  65. if file_name.endswith('_defconfig'):
  66. board_name = file_name[:-len('_defconfig')]
  67. ret[arch].add(Board(board_name, arch, maybe_board))
  68. return ret
  69. def parse_args():
  70. parser = argparse.ArgumentParser()
  71. add_args(parser)
  72. return parser.parse_args()
  73. def add_args(parser):
  74. # Remember to update west-completion.bash if you add or remove
  75. # flags
  76. parser.add_argument("--arch-root", dest='arch_roots', default=[],
  77. type=Path, action='append',
  78. help='''add an architecture root (ZEPHYR_BASE is
  79. always present), may be given more than once''')
  80. parser.add_argument("--board-root", dest='board_roots', default=[],
  81. type=Path, action='append',
  82. help='''add a board root (ZEPHYR_BASE is always
  83. present), may be given more than once''')
  84. def dump_boards(arch2boards):
  85. for arch, boards in arch2boards.items():
  86. print(f'{arch}:')
  87. for board in boards:
  88. print(f' {board.name}')
  89. if __name__ == '__main__':
  90. dump_boards(find_arch2boards(parse_args()))