128 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			128 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import subprocess
 | |
| import pathlib
 | |
| import time
 | |
| 
 | |
| build_format = '| {:29} | {:30} | {:18} | {:7} | {:6} | {:6} |'
 | |
| 
 | |
| SUCCEEDED = "\033[32msucceeded\033[0m"
 | |
| FAILED = "\033[31mfailed\033[0m"
 | |
| SKIPPED = "\033[33mskipped\033[0m"
 | |
| 
 | |
| 
 | |
| def skip_example(example, board):
 | |
|     ex_dir = pathlib.Path('examples/') / example
 | |
|     bsp = pathlib.Path("hw/bsp")
 | |
| 
 | |
|     if (bsp / board / "board.mk").exists():
 | |
|         # board without family
 | |
|         board_dir = bsp / board
 | |
|         family = ""
 | |
|         mk_contents = ""
 | |
|     else:
 | |
|         # board within family
 | |
|         board_dir = list(bsp.glob("*/boards/" + board))
 | |
|         if not board_dir:
 | |
|             # Skip unknown boards
 | |
|             return True
 | |
| 
 | |
|         board_dir = list(board_dir)[0]
 | |
| 
 | |
|         family_dir = board_dir.parent.parent
 | |
|         family = family_dir.name
 | |
| 
 | |
|         # family CMake
 | |
|         family_mk = family_dir / "family.cmake"
 | |
| 
 | |
|         # family.mk
 | |
|         if not family_mk.exists():
 | |
|             family_mk = family_dir / "family.mk"
 | |
| 
 | |
|         mk_contents = family_mk.read_text()
 | |
| 
 | |
|     # Find the mcu, first in family mk then board mk
 | |
|     if "CFG_TUSB_MCU=OPT_MCU_" not in mk_contents:
 | |
|         board_mk = board_dir / "board.cmake"
 | |
|         if not board_mk.exists():
 | |
|             board_mk = board_dir / "board.mk"
 | |
| 
 | |
|         mk_contents = board_mk.read_text()
 | |
| 
 | |
|     for token in mk_contents.split():
 | |
|         if "CFG_TUSB_MCU=OPT_MCU_" in token:
 | |
|             # Strip " because cmake files has them.
 | |
|             token = token.strip("\"")
 | |
|             _, opt_mcu = token.split("=")
 | |
|             mcu = opt_mcu[len("OPT_MCU_"):]
 | |
| 
 | |
|     # Skip all OPT_MCU_NONE these are WIP port
 | |
|     if mcu == "NONE":
 | |
|         return True
 | |
| 
 | |
|     skip_file = ex_dir / "skip.txt"
 | |
|     only_file = ex_dir / "only.txt"
 | |
| 
 | |
|     if skip_file.exists() and only_file.exists():
 | |
|         raise RuntimeError("Only have a skip or only file. Not both.")
 | |
|     elif skip_file.exists():
 | |
|         skips = skip_file.read_text().split()
 | |
|         return ("mcu:" + mcu in skips or
 | |
|                 "board:" + board in skips or
 | |
|                 "family:" + family in skips)
 | |
|     elif only_file.exists():
 | |
|         onlys = only_file.read_text().split()
 | |
|         return not ("mcu:" + mcu in onlys or
 | |
|                     "board:" + board in onlys or
 | |
|                     "family:" + family in onlys)
 | |
| 
 | |
|     return False
 | |
| 
 | |
| 
 | |
| def build_example(example, board, make_option):
 | |
|     start_time = time.monotonic()
 | |
|     flash_size = "-"
 | |
|     sram_size = "-"
 | |
| 
 | |
|     # succeeded, failed, skipped
 | |
|     ret = [0, 0, 0]
 | |
| 
 | |
|     make_cmd = "make -j -C examples/{} BOARD={} {}".format(example, board, make_option)
 | |
| 
 | |
|     # Check if board is skipped
 | |
|     if skip_example(example, board):
 | |
|         status = SKIPPED
 | |
|         ret[2] = 1
 | |
|         print(build_format.format(example, board, status, '-', flash_size, sram_size))
 | |
|     else:
 | |
|         subprocess.run(make_cmd + " clean", shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
 | |
|         build_result = subprocess.run(make_cmd + " all", shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
 | |
| 
 | |
|         if build_result.returncode == 0:
 | |
|             status = SUCCEEDED
 | |
|             ret[0] = 1
 | |
|             (flash_size, sram_size) = build_size(make_cmd)
 | |
|             #subprocess.run(make_cmd + " copy-artifact", shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
 | |
|         else:
 | |
|             status = FAILED
 | |
|             ret[1] = 1
 | |
| 
 | |
|         build_duration = time.monotonic() - start_time
 | |
|         print(build_format.format(example, board, status, "{:.2f}s".format(build_duration), flash_size, sram_size))
 | |
| 
 | |
|         if build_result.returncode != 0:
 | |
|             print(build_result.stdout.decode("utf-8"))
 | |
| 
 | |
|     return ret
 | |
| 
 | |
| 
 | |
| def build_size(make_cmd):
 | |
|     size_output = subprocess.run(make_cmd + ' size', shell=True, stdout=subprocess.PIPE).stdout.decode("utf-8").splitlines()
 | |
|     for i, l in enumerate(size_output):
 | |
|         text_title = 'text	   data	    bss	    dec'
 | |
|         if text_title in l:
 | |
|             size_list = size_output[i+1].split('\t')
 | |
|             flash_size = int(size_list[0])
 | |
|             sram_size = int(size_list[1]) + int(size_list[2])
 | |
|             return (flash_size, sram_size)
 | |
| 
 | |
|     return (0, 0)
 | 
