#!/usr/bin/env python # # Copyright (C) 2018 The Android Open Source Project # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # This script looks through compiled object file (stored human readable text), # and looks for the compile-time constants (added through custom "asm" block). # For example: .ascii ">>OBJECT_ALIGNMENT_MASK $7 $0<<" # # It will transform each such line to #define which is usabe in assembly code. # For example: #define OBJECT_ALIGNMENT_MASK 0x7 # # Usage: make_header.py out/soong/.intermediates/.../asm_defines.o # import argparse import re import sys def convert(input): """Find all defines in the compiler generated assembly and convert them to #define pragmas""" asm_define_re = re.compile(r'">>(\w+) (?:\$|#)?([-0-9]+) (?:\$|#)?(0|1)<<"') asm_defines = asm_define_re.findall(input) if not asm_defines: raise RuntimeError("Failed to find any asm defines in the input") # Convert the found constants to #define pragmas. # In case the C++ compiler decides to reorder the AsmDefinesFor_${name} functions, # we don't want the order of the .h file to change from one compilation to another. # Sorting ensures deterministic order of the #defines. output = [] for name, value, negative_value in sorted(asm_defines): value = int(value) if value < 0 and negative_value == "0": # Overflow - uint64_t constant was pretty printed as negative value. value += 2 ** 64 # Python will use arbitrary precision arithmetic. output.append("#define {0} {1:#x}".format(name, value)) return "\n".join(output) if __name__ == "__main__": parser = argparse.ArgumentParser() parser.add_argument('input', help="Object file as text") args = parser.parse_args() print(convert(open(args.input, "r").read()))