| #!/usr/bin/env python3 |
| # SPDX-License-Identifier: GPL-2.0-or-later |
| |
| import os, sys, xml.etree.ElementTree |
| |
| def writeliteral(indent, bytes): |
| sys.stdout.write(' ' * indent) |
| sys.stdout.write('"') |
| quoted = True |
| |
| for c in bytes: |
| if not quoted: |
| sys.stdout.write('\n') |
| sys.stdout.write(' ' * indent) |
| sys.stdout.write('"') |
| quoted = True |
| |
| if c == b'"'[0]: |
| sys.stdout.write('\\"') |
| elif c == b'\\'[0]: |
| sys.stdout.write('\\\\') |
| elif c == b'\n'[0]: |
| sys.stdout.write('\\n"') |
| quoted = False |
| elif c >= 32 and c < 127: |
| sys.stdout.write(c.to_bytes(1, 'big').decode()) |
| else: |
| sys.stdout.write(f'\{c:03o}') |
| |
| if quoted: |
| sys.stdout.write('"') |
| |
| sys.stdout.write('#include "qemu/osdep.h"\n' \ |
| '#include "exec/gdbstub.h"\n' \ |
| '\n' |
| 'const GDBFeature gdb_static_features[] = {\n') |
| |
| for input in sys.argv[1:]: |
| with open(input, 'rb') as file: |
| read = file.read() |
| |
| parser = xml.etree.ElementTree.XMLPullParser(['start', 'end']) |
| parser.feed(read) |
| events = parser.read_events() |
| event, element = next(events) |
| if event != 'start': |
| sys.stderr.write(f'unexpected event: {event}\n') |
| exit(1) |
| if element.tag != 'feature': |
| sys.stderr.write(f'unexpected start tag: {element.tag}\n') |
| exit(1) |
| |
| feature_name = element.attrib['name'] |
| regnum = 0 |
| regnames = [] |
| regnums = [] |
| tags = ['feature'] |
| for event, element in events: |
| if event == 'end': |
| if element.tag != tags[len(tags) - 1]: |
| sys.stderr.write(f'unexpected end tag: {element.tag}\n') |
| exit(1) |
| |
| tags.pop() |
| if element.tag == 'feature': |
| break |
| elif event == 'start': |
| if len(tags) < 2 and element.tag == 'reg': |
| if 'regnum' in element.attrib: |
| regnum = int(element.attrib['regnum']) |
| |
| regnames.append(element.attrib['name']) |
| regnums.append(regnum) |
| regnum += 1 |
| |
| tags.append(element.tag) |
| else: |
| raise Exception(f'unexpected event: {event}\n') |
| |
| if len(tags): |
| sys.stderr.write('unterminated feature tag\n') |
| exit(1) |
| |
| base_reg = min(regnums) |
| num_regs = max(regnums) - base_reg + 1 if len(regnums) else 0 |
| |
| sys.stdout.write(' {\n') |
| writeliteral(8, bytes(os.path.basename(input), 'utf-8')) |
| sys.stdout.write(',\n') |
| writeliteral(8, read) |
| sys.stdout.write(',\n') |
| writeliteral(8, bytes(feature_name, 'utf-8')) |
| sys.stdout.write(',\n (const char * const []) {\n') |
| |
| for index, regname in enumerate(regnames): |
| sys.stdout.write(f' [{regnums[index] - base_reg}] =\n') |
| writeliteral(16, bytes(regname, 'utf-8')) |
| sys.stdout.write(',\n') |
| |
| sys.stdout.write(f' }},\n {num_regs},\n }},\n') |
| |
| sys.stdout.write(' { NULL }\n};\n') |