mirror of https://github.com/zeldaret/tmc.git
				
				
				
			
		
			
				
	
	
		
			154 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
			
		
		
	
	
			154 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
| #!/usr/bin/env python3
 | |
| 
 | |
| import argparse
 | |
| import json
 | |
| import git
 | |
| import os
 | |
| import re
 | |
| 
 | |
| def collect_non_matching_funcs():
 | |
|     result = []
 | |
|     for root, dirs, files in os.walk('src'):
 | |
|         for file in files:
 | |
|             if file.endswith('.c'):
 | |
|                 with open(os.path.join(root, file), 'r') as f:
 | |
|                     data = f.read()
 | |
|                     # Find all NONMATCH and ASM_FUNC macros
 | |
|                     for match in re.findall(r'(NONMATCH|ASM_FUNC)\(".*",\W*\w*\W*(\w*).*\)', data):
 | |
|                         result.append(match)
 | |
|     return result
 | |
| 
 | |
| 
 | |
| def parse_map(non_matching_funcs):
 | |
|     src = 0
 | |
|     asm = 0
 | |
|     src_data = 0
 | |
|     data = 0
 | |
|     non_matching = 0
 | |
| 
 | |
|     with open('tmc.map', 'r') as map:
 | |
|         # Skip to the linker script section
 | |
|         line = map.readline()
 | |
|         while not line.startswith('Linker script and memory map'):
 | |
|             line = map.readline()
 | |
|         while not line.startswith('rom'):
 | |
|             line = map.readline()
 | |
| 
 | |
|         prev_symbol = None
 | |
|         prev_addr = 0
 | |
|         for line in map:
 | |
|             if line.startswith(' .'):
 | |
|                 arr = line.split()
 | |
|                 section = arr[0]
 | |
|                 size = int(arr[2], 16)
 | |
|                 filepath = arr[3]
 | |
|                 dir = filepath.split('/')[0]
 | |
| 
 | |
|                 if section == '.text':
 | |
|                     if dir == 'src':
 | |
|                         src += size
 | |
|                     elif dir == 'asm':
 | |
|                         if filepath.find("asm/src/") != -1 or filepath.find("asm/lib/") != -1:
 | |
|                             src += size
 | |
|                         else:
 | |
|                             asm += size
 | |
|                     elif dir == 'data':
 | |
|                         # scripts
 | |
|                         src_data += size
 | |
|                     elif dir == '..':
 | |
|                         # libc
 | |
|                         src += size
 | |
|                 elif section == '.rodata':
 | |
|                     if dir == 'src':
 | |
|                         src_data += size
 | |
|                     elif dir == 'data':
 | |
|                         data += size
 | |
| 
 | |
|             elif line.startswith('  '):
 | |
|                 arr = line.split()
 | |
|                 if len(arr) == 2 and arr[1] != '':  # It is actually a symbol
 | |
| 
 | |
|                     if prev_symbol in non_matching_funcs:
 | |
|                         # Calculate the length for non matching function
 | |
|                         non_matching += int(arr[0], 16) - prev_addr
 | |
| 
 | |
|                     prev_symbol = arr[1]
 | |
|                     prev_addr = int(arr[0], 16)
 | |
|             elif line.strip() == '':
 | |
|                 # End of linker script section
 | |
|                 break
 | |
| 
 | |
|     src -= non_matching
 | |
|     asm += non_matching
 | |
| 
 | |
|     return (src, asm, src_data, data)
 | |
| 
 | |
| 
 | |
| def main():
 | |
|     parser = argparse.ArgumentParser()
 | |
| 
 | |
|     parser = argparse.ArgumentParser(description="Computes current progress throughout the whole project.")
 | |
|     parser.add_argument("format", nargs="?", default="text", choices=["text", "csv", "shield-json"])
 | |
|     parser.add_argument("-m", "--matching", dest='matching', action='store_true',
 | |
|                         help="Output matching progress instead of decompilation progress")
 | |
|     args = parser.parse_args()
 | |
| 
 | |
|     matching = args.matching
 | |
| 
 | |
|     non_matching_funcs = []
 | |
|     funcs = collect_non_matching_funcs()
 | |
|     if matching:
 | |
|         # Remove all non matching funcs from count
 | |
|         non_matching_funcs = [x[1] for x in funcs]
 | |
|     else:
 | |
|         # Only remove ASM_FUNC functions from count
 | |
|         for func in funcs:
 | |
|             if func[0] == 'ASM_FUNC':
 | |
|                 non_matching_funcs.append(func[1])
 | |
| 
 | |
|     (src, asm, src_data, data) = parse_map(non_matching_funcs)
 | |
| 
 | |
|     total = src + asm
 | |
|     data_total = src_data + data
 | |
| 
 | |
|     src_percent = 100 * src / total
 | |
|     asm_percent = 100 * asm / total
 | |
| 
 | |
|     src_data_percent = 100 * src_data / data_total
 | |
|     data_percent     = 100 * data / data_total
 | |
| 
 | |
| 
 | |
|     if args.format == 'csv':
 | |
|         version = 2
 | |
|         git_object = git.Repo().head.object
 | |
|         timestamp = str(git_object.committed_date)
 | |
|         git_hash = git_object.hexsha
 | |
| 
 | |
|         csv_list = [str(version), timestamp, git_hash, str(src),
 | |
|                     str(total), str(src_data), str(data_total)]
 | |
| 
 | |
|         print(','.join(csv_list))
 | |
| 
 | |
|     elif args.format == 'shield-json':
 | |
|         # https://shields.io/endpoint
 | |
|         print(json.dumps({
 | |
|             "schemaVersion": 1,
 | |
|             "label": "progress",
 | |
|             "message": f"{src_percent:.3g}%",
 | |
|             "color": 'yellow',
 | |
|         }))
 | |
| 
 | |
|     elif args.format == 'text':
 | |
|         adjective = "decompiled" if not args.matching else "matched"
 | |
| 
 | |
|         print("src:  {:>9} / {:>8} total bytes {:<10} {:>9.4f}%".format(src, total, adjective, round(src_percent, 4)))
 | |
|         # print()
 | |
|         print("data: {:>9} / {:>8} total bytes analysed   {:>9.4f}%".format(src_data, data_total, round(src_data_percent, 4)))
 | |
| 
 | |
|     else:
 | |
|         print("Unknown format argument: " + args.format)
 | |
| 
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     main()
 |