1abaf1209SJerome Forissier#!/usr/bin/env python3 2abaf1209SJerome Forissier# 3abaf1209SJerome Forissier# Copyright (c) 2014-2017, Linaro Limited 4abaf1209SJerome Forissier# All rights reserved. 5abaf1209SJerome Forissier# 6abaf1209SJerome Forissier# SPDX-License-Identifier: BSD-3-Clause 7abaf1209SJerome Forissier 8abaf1209SJerome Forissierimport argparse 9abaf1209SJerome Forissierimport os 10abaf1209SJerome Forissierimport subprocess 11abaf1209SJerome Forissierimport sys 12abaf1209SJerome Forissier 13abaf1209SJerome Forissier 14abaf1209SJerome Forissierdef get_args(): 153be7773fSJerome Forissier parser = argparse.ArgumentParser(description='Shows the memory usage ' 163be7773fSJerome Forissier 'of an OP-TEE based on ELF sections') 173be7773fSJerome Forissier parser.add_argument('tee_elf', help='the OP-TEE ELF file (tee.elf)') 183be7773fSJerome Forissier parser.add_argument('-a', '--all', action='store_true', 193be7773fSJerome Forissier help=' same as -i -p -u -U') 203be7773fSJerome Forissier parser.add_argument('-n', '--no-map', action='store_true', 213be7773fSJerome Forissier help=' do not show the detailed section mappings and ' 223be7773fSJerome Forissier 'RAM usage') 233be7773fSJerome Forissier parser.add_argument('-i', '--init', action='store_true', 243be7773fSJerome Forissier help='report the total size of the .*_init sections') 253be7773fSJerome Forissier parser.add_argument('-p', '--paged', action='store_true', 263be7773fSJerome Forissier help='report the total size of the .*_pageable ' 273be7773fSJerome Forissier 'sections') 283be7773fSJerome Forissier parser.add_argument('-u', '--unpaged', action='store_true', 293be7773fSJerome Forissier help='report the total size of the unpaged sections, ' 303be7773fSJerome Forissier 'that is, all sections but the ones in --init or ' 313be7773fSJerome Forissier '--paged') 323be7773fSJerome Forissier parser.add_argument('-U', '--unpaged-no-heap', action='store_true', 333be7773fSJerome Forissier help='report the size of all unpaged sections ' 343be7773fSJerome Forissier 'excluding heap space. Reflects the size of unpaged ' 353be7773fSJerome Forissier 'code and data (.text, .rodata, .data, .bss, .nozi ' 363be7773fSJerome Forissier 'and possibly unwind tables)') 373be7773fSJerome Forissier parser.add_argument('-r', '--raw', action='store_true', 383be7773fSJerome Forissier help='when processing -i, -p, -u, or -U, show only ' 393be7773fSJerome Forissier 'the size (in decimal) and no other text') 40abaf1209SJerome Forissier return parser.parse_args() 41abaf1209SJerome Forissier 42abaf1209SJerome Forissier 43abaf1209SJerome Forissierdef printf(format, *args): 44abaf1209SJerome Forissier sys.stdout.write(format % args) 45abaf1209SJerome Forissier 46abaf1209SJerome Forissier 47abaf1209SJerome Forissierdef print_sect(name, addr, size, round_up=False, print_num_pages=False): 483be7773fSJerome Forissier if args.no_map: 493be7773fSJerome Forissier return 50abaf1209SJerome Forissier if size == 0: 51abaf1209SJerome Forissier size_kib = 0 52abaf1209SJerome Forissier num_pages = 0 53abaf1209SJerome Forissier else: 54abaf1209SJerome Forissier if round_up: 55abaf1209SJerome Forissier size_kib = (size - 1) / 1024 + 1 56abaf1209SJerome Forissier else: 57abaf1209SJerome Forissier size_kib = size / 1024 58abaf1209SJerome Forissier num_pages = (size - 1) / 4096 + 1 59abaf1209SJerome Forissier 60abaf1209SJerome Forissier printf('%-16s %.8X - %.8X size %.8X %3d KiB', name, addr, addr + size, 61abaf1209SJerome Forissier size, size_kib) 62abaf1209SJerome Forissier if print_num_pages: 63abaf1209SJerome Forissier printf(' %d pages', num_pages) 64abaf1209SJerome Forissier printf('\n') 65abaf1209SJerome Forissier 66abaf1209SJerome Forissier 673be7773fSJerome Forissierdef print_pager_stat(name, size): 683be7773fSJerome Forissier size_kib = size / 1024 693be7773fSJerome Forissier if args.raw: 703be7773fSJerome Forissier printf('%d ', size) 713be7773fSJerome Forissier else: 723be7773fSJerome Forissier printf('%-36s size %.8X %3d KiB\n', name, size, size_kib) 733be7773fSJerome Forissier 743be7773fSJerome Forissier 75abaf1209SJerome Forissierdef readelf_cmd(): 76abaf1209SJerome Forissier return os.getenv('CROSS_COMPILE', '') + 'readelf' 77abaf1209SJerome Forissier 78abaf1209SJerome Forissier 79abaf1209SJerome Forissierdef main(): 803be7773fSJerome Forissier global args 813be7773fSJerome Forissier 82abaf1209SJerome Forissier in_shdr = False 83abaf1209SJerome Forissier sects = [] 843be7773fSJerome Forissier init_size = 0 853be7773fSJerome Forissier paged_size = 0 863be7773fSJerome Forissier unpaged_size = 0 873be7773fSJerome Forissier unpaged_no_heap_size = 0 88abaf1209SJerome Forissier 89abaf1209SJerome Forissier args = get_args() 90abaf1209SJerome Forissier env = os.environ.copy() 91abaf1209SJerome Forissier env['LC_ALL'] = 'C' 92*edbeddc1SJerome Forissier readelf = subprocess.Popen(str.split(readelf_cmd()) + ['-S', '-W', 93*edbeddc1SJerome Forissier args.tee_elf], 94abaf1209SJerome Forissier stdout=subprocess.PIPE, env=env, 95abaf1209SJerome Forissier universal_newlines=True) 96abaf1209SJerome Forissier for line in iter(readelf.stdout.readline, ''): 97abaf1209SJerome Forissier if 'Section Headers:' in line: 98abaf1209SJerome Forissier in_shdr = True 99abaf1209SJerome Forissier continue 100abaf1209SJerome Forissier if 'Key to Flags:' in line: 101abaf1209SJerome Forissier in_shdr = False 102abaf1209SJerome Forissier continue 103abaf1209SJerome Forissier if in_shdr: 104abaf1209SJerome Forissier words = line.split() 105abaf1209SJerome Forissier if words[0] == '[': 106abaf1209SJerome Forissier words.pop(0) 107abaf1209SJerome Forissier try: 108abaf1209SJerome Forissier (_, name, _, addr, offs, size, _, 109abaf1209SJerome Forissier flags) = words[:8] 110abaf1209SJerome Forissier except: 111abaf1209SJerome Forissier continue 112abaf1209SJerome Forissier if (flags == 'AX' or flags == 'WA' or flags == 'A' or 113abaf1209SJerome Forissier flags == 'AL'): 114abaf1209SJerome Forissier sects.append({'name': name, 'addr': addr, 115abaf1209SJerome Forissier 'offs': offs, 'size': size}) 116abaf1209SJerome Forissier for sect in sects: 117abaf1209SJerome Forissier if sect['addr'] != 0: 118abaf1209SJerome Forissier first_addr = sect['addr'] 119abaf1209SJerome Forissier break 120abaf1209SJerome Forissier last_addr = sects[-1]['addr'] 121abaf1209SJerome Forissier last_size = sects[-1]['size'] 122abaf1209SJerome Forissier 123abaf1209SJerome Forissier ram_usage = int(last_addr, 16) + int(last_size, 16) - int(first_addr, 16) 124abaf1209SJerome Forissier print_sect('RAM Usage', int(first_addr, 16), ram_usage, True, True) 125abaf1209SJerome Forissier 126abaf1209SJerome Forissier last_addr = 0 127abaf1209SJerome Forissier last_size = 0 128abaf1209SJerome Forissier for sect in sects: 129abaf1209SJerome Forissier name = sect['name'] 130abaf1209SJerome Forissier addr = int(sect['addr'], 16) 131abaf1209SJerome Forissier size = int(sect['size'], 16) 132abaf1209SJerome Forissier 133abaf1209SJerome Forissier if last_addr != 0 and addr != last_addr + last_size: 134abaf1209SJerome Forissier print_sect('*hole*', last_addr + last_size, 135abaf1209SJerome Forissier addr - (last_addr + last_size)) 136abaf1209SJerome Forissier print_sect(name, addr, size) 1373be7773fSJerome Forissier if name.endswith('_init'): 1383be7773fSJerome Forissier init_size += size 1393be7773fSJerome Forissier elif name.endswith('_pageable'): 1403be7773fSJerome Forissier paged_size += size 1413be7773fSJerome Forissier else: 1423be7773fSJerome Forissier if not name.startswith('.heap'): 1433be7773fSJerome Forissier unpaged_no_heap_size += size 1443be7773fSJerome Forissier unpaged_size += size 145abaf1209SJerome Forissier last_addr = addr 146abaf1209SJerome Forissier last_size = size 147abaf1209SJerome Forissier 1483be7773fSJerome Forissier if args.all or args.init: 1493be7773fSJerome Forissier print_pager_stat('Init sections (.*_init)', init_size) 1503be7773fSJerome Forissier if args.all or args.paged: 1513be7773fSJerome Forissier print_pager_stat('Paged sections (.*_pageable)', paged_size) 1523be7773fSJerome Forissier if args.all or args.unpaged: 1533be7773fSJerome Forissier print_pager_stat('Unpaged sections ', unpaged_size) 1543be7773fSJerome Forissier if args.all or args.unpaged_no_heap: 1553be7773fSJerome Forissier print_pager_stat('Unpaged sections (heap excluded)', 1563be7773fSJerome Forissier unpaged_no_heap_size) 1573be7773fSJerome Forissier if (args.raw and (args.all or args.init or args.paged or 1583be7773fSJerome Forissier args.unpaged or args.unpaged_no_heap)): 1593be7773fSJerome Forissier printf('\n') 1603be7773fSJerome Forissier 161abaf1209SJerome Forissier 162abaf1209SJerome Forissierif __name__ == "__main__": 163abaf1209SJerome Forissier main() 164