| #!/usr/bin/env python3 |
| |
| import contextlib |
| import os |
| import re |
| import sys |
| |
| from urllib.request import urlopen |
| |
| PROJECT_ROOT = os.path.dirname(os.path.dirname(os.path.realpath(__file__))) |
| SELF_PATH = os.path.relpath(__file__, PROJECT_ROOT).replace('\\', '/') |
| |
| syscalls = {} |
| archs = ['arm64', 'arm32', 'x86', 'x86_64'] |
| |
| |
| def print_tables(): |
| print('// DO NOT EDIT. Auto-generated by %s' % SELF_PATH) |
| print('#ifndef SRC_KERNEL_UTILS_SYSCALL_TABLE_GENERATED_H_') |
| print('#define SRC_KERNEL_UTILS_SYSCALL_TABLE_GENERATED_H_') |
| print('') |
| print('#include <stdint.h>') |
| print('') |
| print('#include "src/kernel_utils/syscall_table.h"') |
| print('') |
| print('namespace perfetto {') |
| print('') |
| # First of all merge all the syscall names from all tables and emit one big |
| # string with unique names. |
| syscall_names = set() |
| for arch in archs: |
| tab = syscalls[arch] |
| for i in range(max(tab.keys()) + 1): |
| syscall_names.add(tab.get(i, '')) |
| print('constexpr char kAllSyscalllNames[] =') |
| offset_by_name = {} |
| last_off = 0 |
| for syscall_name in sorted(syscall_names): |
| if syscall_name in offset_by_name: |
| continue |
| print(' "%s\\0"' % syscall_name) |
| offset_by_name[syscall_name] = last_off |
| last_off += len(syscall_name) + 1 |
| if last_off >= 0xffff: |
| raise Exception('SyscallTable::OffT must be increased to uint32_t') |
| print(';\n') |
| # Then print the individual tables. |
| for arch in archs: |
| tab = syscalls[arch] |
| print('struct SyscallTable_%s {' % arch) |
| print(' static constexpr const char* names = kAllSyscalllNames;') |
| print(' static constexpr SyscallTable::OffT offsets[] {') |
| for i in range(max(tab.keys()) + 1): |
| syscall_name = tab.get(i, '') |
| print('%d, // %d: %s' % (offset_by_name[syscall_name], i, syscall_name)) |
| print(' };') |
| print('};\n') |
| print('} // namespace perfetto') |
| print('#endif // SRC_KERNEL_UTILS_SYSCALL_TABLE_GENERATED_H_') |
| |
| |
| # Parses a .tbl file (new format). |
| def parse_tlb(data): |
| table = {} |
| for line in data.splitlines(): |
| line = line.strip() |
| if line.startswith('#') or not (line): |
| continue |
| parts = line.split() |
| table[int(parts[0])] = 'sys_' + parts[2] |
| return table |
| |
| |
| # Parses a #define __NR_xx 1234 old-style unistd.h header. |
| def parse_def(data): |
| table = {} |
| for line in data.splitlines(): |
| m = re.match(r'^#define\s+__NR\d*?_(\w+)\s+(\d+)\s*$', line.strip()) |
| if not m or m.group(1) == 'syscalls': # __NR_syscalls is just a sentinel. |
| continue |
| table[int(m.group(2))] = 'sys_' + m.group(1) |
| return table |
| |
| |
| def Main(): |
| KSRC = 'https://raw.githubusercontent.com/torvalds/linux/v6.7/' |
| |
| response = urlopen(KSRC + 'arch/x86/entry/syscalls/syscall_64.tbl') |
| syscalls['x86_64'] = parse_tlb(response.read().decode()) |
| |
| response = urlopen(KSRC + 'arch/x86/entry/syscalls/syscall_32.tbl') |
| syscalls['x86'] = parse_tlb(response.read().decode()) |
| |
| response = urlopen(KSRC + 'arch/arm/tools/syscall.tbl') |
| syscalls['arm32'] = parse_tlb(response.read().decode()) |
| |
| # From: |
| # arch/arm64/include/asm/unistd.h |
| # -> arch/arm64/include/uapi/asm/unistd.h |
| # -> include/uapi/asm-generic/unistd.h |
| response = urlopen(KSRC + 'include/uapi/asm-generic/unistd.h') |
| syscalls['arm64'] = parse_def(response.read().decode()) |
| |
| dst_file = os.path.join(PROJECT_ROOT, 'src', 'kernel_utils', |
| 'syscall_table_generated.h') |
| tmp_file = dst_file + '.tmp' |
| |
| print('Writing ', dst_file) |
| with open(tmp_file, 'w') as f: |
| with contextlib.redirect_stdout(f): |
| print_tables() |
| os.rename(tmp_file, dst_file) |
| |
| print('Running clang-format (might fail if depot_tools isn\'t in the PATH)') |
| os.system('clang-format -i ' + dst_file) |
| |
| if __name__ == '__main__': |
| sys.exit(Main()) |