xref: /optee_os/scripts/mem_usage.py (revision edbeddc184ead93b125e49d55c33644d6d2762e9)
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