build-gem5 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. #!/usr/bin/env python3
  2. import os
  3. import pathlib
  4. import subprocess
  5. import tempfile
  6. import common
  7. from shell_helpers import LF
  8. class Main(common.BuildCliFunction):
  9. def __init__(self):
  10. super().__init__(
  11. description='''\
  12. Build gem5.
  13. https://github.com/cirosantilli/linux-kernel-module-cheat-regression#gem5-buildroot-setup
  14. '''
  15. )
  16. self.add_argument(
  17. '--unit-test',
  18. action='append',
  19. default=[],
  20. help='''\
  21. Build and run the given unit test. Paths are relative to src/ without the .opt suffix.
  22. If given multiple times, runs multiple unit tests. Ignore --unit-tests.
  23. https://github.com/cirosantilli/linux-kernel-module-cheat-regression#gem5-unit-tests
  24. '''
  25. )
  26. self.add_argument(
  27. '--unit-tests',
  28. default=False,
  29. help='''\
  30. Build and run all the gem5 unit tests instead of the gem5 executable.
  31. https://github.com/cirosantilli/linux-kernel-module-cheat-regression#gem5-unit-tests
  32. '''
  33. )
  34. self._add_argument('--ldflags')
  35. self._add_argument('extra_make_args')
  36. self.eclipse_tmpdir = None
  37. def build(self):
  38. build_dir = self.get_build_dir()
  39. binaries_dir = self.env['gem5_system_binaries_dir']
  40. disks_dir = os.path.join(self.env['gem5_system_dir'], 'disks')
  41. os.makedirs(binaries_dir, exist_ok=True)
  42. os.makedirs(disks_dir, exist_ok=True)
  43. if not os.path.exists(os.path.join(self.env['gem5_source_dir'], '.git')):
  44. if self.env['_args_given']['gem5_worktree']:
  45. self.sh.run_cmd([
  46. 'git', LF,
  47. '-C', self.env['gem5_default_source_dir'], LF,
  48. 'worktree', 'add', LF,
  49. '-b', os.path.join('wt', self.env['gem5_worktree']), LF,
  50. self.env['gem5_source_dir'], LF,
  51. ])
  52. else:
  53. if not self.env['dry_run']:
  54. raise Exception('gem5 submodule not checked out')
  55. if self.env['verbose']:
  56. verbose = ['--verbose', LF]
  57. else:
  58. verbose = []
  59. if self.env['is_arm']:
  60. gem5_system_source_dir = os.path.join(self.env['gem5_source_dir'], 'system')
  61. # dtb
  62. dt_source_dir = os.path.join(gem5_system_source_dir, 'arm', 'dt')
  63. dt_build_dir = os.path.join(self.env['gem5_system_dir'], 'arm', 'dt')
  64. self.sh.run_cmd(['make', '-C', dt_source_dir, LF])
  65. self.sh.copy_dir_if_update_non_recursive(
  66. srcdir=dt_source_dir,
  67. destdir=dt_build_dir,
  68. filter_ext='.dtb',
  69. )
  70. # Bootloader 32.
  71. arm_bootloader_dir = os.path.join(gem5_system_source_dir, 'arm', 'bootloader')
  72. bootloader32_dir = os.path.join(arm_bootloader_dir, 'arm')
  73. # TODO use the buildroot cross compiler here, and remove the dependencies from configure.
  74. self.sh.run_cmd([
  75. 'make', LF,
  76. '-C', bootloader32_dir, LF,
  77. 'CROSS_COMPILE=arm-linux-gnueabihf-', LF,
  78. ])
  79. # bootloader
  80. self.sh.cp(os.path.join(bootloader32_dir, 'boot.arm'), binaries_dir)
  81. # Bootloader 64.
  82. bootloader64_dir = os.path.join(arm_bootloader_dir, 'arm64')
  83. # TODO cross_compile is ignored because the make does not use CC...
  84. self.sh.run_cmd(['make', '-C', bootloader64_dir, LF])
  85. self.sh.cp(os.path.join(bootloader64_dir, 'boot.arm64'), binaries_dir)
  86. self.sh.cp(os.path.join(bootloader64_dir, 'boot_v2.arm64'), binaries_dir)
  87. term_source_dir = os.path.join(self.env['gem5_source_dir'], 'util/term')
  88. m5term_build = os.path.join(term_source_dir, 'm5term')
  89. self.sh.run_cmd(['make', '-C', term_source_dir, LF])
  90. if os.path.exists(self.env['gem5_m5term']):
  91. # Otherwise self.sh.cp would fail with "Text file busy" if you
  92. # tried to rebuild while running m5term:
  93. # https://stackoverflow.com/questions/16764946/what-generates-the-text-file-busy-message-in-unix/52427512#52427512
  94. self.sh.rmrf(self.env['gem5_m5term'])
  95. self.sh.cp(m5term_build, self.env['gem5_m5term'])
  96. if self.env['unit_test']:
  97. targets = [self.get_gem5_target_path(self.env, test) for test in self.env['unit_test']]
  98. elif self.env['unit_tests']:
  99. targets = [self.env['gem5_unit_test_target']]
  100. else:
  101. targets = [self.env['gem5_executable']]
  102. if self.env['gem5_clang']:
  103. extra_env = {
  104. 'CC': 'clang',
  105. 'CXX': 'clang++',
  106. }
  107. else:
  108. extra_env = {}
  109. # https://cirosantilli.com/cirodown#benchmark-gem5-single-file-change-rebuild-time
  110. ldflags_extra = ['-fuse-ld=lld'] + self.env['ldflags']
  111. kwargs = {}
  112. if self.env['ccache']:
  113. kwargs['extra_paths'] = [self.env['ccache_dir']]
  114. exit_status = self.sh.run_cmd(
  115. (
  116. [
  117. 'scons', LF,
  118. '-j', str(self.env['nproc']), LF,
  119. '--ignore-style', LF,
  120. 'LDFLAGS_EXTRA={}'.format(self.sh.cmd_to_string(ldflags_extra, force_oneline=True)), LF,
  121. 'USE_HDF5=1', LF,
  122. ] +
  123. verbose +
  124. [
  125. # TODO reenable, broken, had enough of this.
  126. # https://gem5.atlassian.net/browse/GEM5-357
  127. # https://gem5.atlassian.net/browse/GEM5-656
  128. # https://gem5.atlassian.net/browse/GEM5-778
  129. #'SLICC_HTML=True', LF,
  130. ] +
  131. self.sh.add_newlines(targets) +
  132. self.sh.add_newlines(self.env['extra_make_args'])
  133. ),
  134. cwd=self.env['gem5_source_dir'],
  135. extra_env=extra_env,
  136. raise_on_failure = False,
  137. **kwargs
  138. )
  139. return exit_status
  140. def clean_pre(self, builddir):
  141. if os.path.exists(self.env['gem5_eclipse_cproject_path']):
  142. self.eclipse_tmpdir = tempfile.mkdtemp()
  143. self.sh.mv(self.env['gem5_eclipse_cproject_path'], self.eclipse_tmpdir)
  144. self.sh.mv(self.env['gem5_eclipse_project_path'], self.eclipse_tmpdir)
  145. def clean_post(self, builddir):
  146. if self.eclipse_tmpdir is not None:
  147. self.sh.mkdir_p(self.env['gem5_build_build_dir'])
  148. self.sh.mv(os.path.join(self.eclipse_tmpdir, self.env['gem5_eclipse_cproject_basename']), self.env['gem5_build_build_dir'])
  149. self.sh.mv(os.path.join(self.eclipse_tmpdir, self.env['gem5_eclipse_project_basename']), self.env['gem5_build_build_dir'])
  150. self.sh.rmrf(self.eclipse_tmpdir)
  151. def get_build_dir(self):
  152. return self.env['gem5_build_dir']
  153. if __name__ == '__main__':
  154. Main().cli()