#!/usr/bin/env python3 # Unicorn Engine # By Dang Hoang Vu, 2013 from __future__ import print_function import sys, re, os INCL_DIR = os.path.join('..', 'include', 'unicorn') include = [ 'arm.h', 'arm64.h', 'mips.h', 'x86.h', 'sparc.h', 'm68k.h', 'ppc.h', 'unicorn.h' ] template = { 'python': { 'header': "# For Unicorn Engine. AUTO-GENERATED FILE, DO NOT EDIT [%s_const.py]\n", 'footer': "", 'line_format': 'UC_%s = %s\n', 'out_file': './python/unicorn/%s_const.py', # prefixes for constant filenames of all archs - case sensitive 'arm.h': 'arm', 'arm64.h': 'arm64', 'mips.h': 'mips', 'x86.h': 'x86', 'sparc.h': 'sparc', 'm68k.h': 'm68k', 'ppc.h': 'ppc', 'unicorn.h': 'unicorn', 'comment_open': '#', 'comment_close': '', }, 'ruby': { 'header': "# For Unicorn Engine. AUTO-GENERATED FILE, DO NOT EDIT [%s_const.rb]\n\nmodule UnicornEngine\n", 'footer': "end", 'line_format': '\tUC_%s = %s\n', 'out_file': './ruby/unicorn_gem/lib/unicorn_engine/%s_const.rb', # prefixes for constant filenames of all archs - case sensitive 'arm.h': 'arm', 'arm64.h': 'arm64', 'mips.h': 'mips', 'x86.h': 'x86', 'sparc.h': 'sparc', 'm68k.h': 'm68k', 'ppc.h': 'ppc', 'unicorn.h': 'unicorn', 'comment_open': '#', 'comment_close': '', }, 'go': { 'header': "package unicorn\n// For Unicorn Engine. AUTO-GENERATED FILE, DO NOT EDIT [%s_const.go]\nconst (\n", 'footer': ")", 'line_format': '\t%s = %s\n', 'out_file': './go/unicorn/%s_const.go', # prefixes for constant filenames of all archs - case sensitive 'arm.h': 'arm', 'arm64.h': 'arm64', 'mips.h': 'mips', 'x86.h': 'x86', 'sparc.h': 'sparc', 'm68k.h': 'm68k', 'ppc.h': 'ppc', 'unicorn.h': 'unicorn', 'comment_open': '//', 'comment_close': '', }, 'java': { 'header': "// For Unicorn Engine. AUTO-GENERATED FILE, DO NOT EDIT\n\npackage unicorn;\n\npublic interface %sConst {\n", 'footer': "\n}\n", 'line_format': ' public static final int UC_%s = %s;\n', 'out_file': './java/unicorn/%sConst.java', # prefixes for constant filenames of all archs - case sensitive 'arm.h': 'Arm', 'arm64.h': 'Arm64', 'mips.h': 'Mips', 'x86.h': 'X86', 'sparc.h': 'Sparc', 'm68k.h': 'M68k', 'ppc.h': 'ppc', 'unicorn.h': 'Unicorn', 'comment_open': '//', 'comment_close': '', }, 'dotnet': { 'header': "// For Unicorn Engine. AUTO-GENERATED FILE, DO NOT EDIT\n\nnamespace UnicornManaged.Const\n\nopen System\n\n[]\nmodule %s =\n", 'footer': "\n", 'line_format': ' let UC_%s = %s\n', 'out_file': os.path.join('dotnet', 'UnicornManaged', 'Const', '%s.fs'), # prefixes for constant filenames of all archs - case sensitive 'arm.h': 'Arm', 'arm64.h': 'Arm64', 'mips.h': 'Mips', 'x86.h': 'X86', 'sparc.h': 'Sparc', 'm68k.h': 'M68k', 'ppc.h': 'ppc', 'unicorn.h': 'Common', 'comment_open': ' //', 'comment_close': '', }, 'pascal': { 'header': "// For Unicorn Engine. AUTO-GENERATED FILE, DO NOT EDIT\n\nunit %sConst;\n\ninterface\n\nconst", 'footer': "\nimplementation\nend.", 'line_format': ' UC_%s = %s;\n', 'out_file': os.path.join('pascal', 'unicorn', '%sConst.pas'), # prefixes for constant filenames of all archs - case sensitive 'arm.h': 'Arm', 'arm64.h': 'Arm64', 'mips.h': 'Mips', 'x86.h': 'X86', 'sparc.h': 'Sparc', 'm68k.h': 'M68k', 'ppc.h': 'ppc', 'unicorn.h': 'Unicorn', 'comment_open': '//', 'comment_close': '', }, } # markup for comments to be added to autogen files MARKUP = '//>' def gen(lang): global include, INCL_DIR templ = template[lang] for target in include: prefix = templ[target] outfile = open(templ['out_file'] %(prefix), 'wb') # open as binary prevents windows newlines outfile.write((templ['header'] % (prefix)).encode("utf-8")) if target == 'unicorn.h': prefix = '' with open(os.path.join(INCL_DIR, target)) as f: lines = f.readlines() previous = {} count = 0 for line in lines: line = line.strip() if line.startswith(MARKUP): # markup for comments outfile.write(("\n%s%s%s\n" %(templ['comment_open'], \ line.replace(MARKUP, ''), templ['comment_close'])).encode("utf-8")) continue if line == '' or line.startswith('//'): continue tmp = line.strip().split(',') for t in tmp: t = t.strip() if not t or t.startswith('//'): continue f = re.split('\s+', t) # parse #define UC_TARGET (num) define = False if f[0] == '#define' and len(f) >= 3: define = True f.pop(0) f.insert(1, '=') if f[0].startswith("UC_" + prefix.upper()): if len(f) > 1 and f[1] not in ('//', '='): print("WARNING: Unable to convert %s" % f) print(" Line =", line) continue elif len(f) > 1 and f[1] == '=': rhs = ''.join(f[2:]) else: rhs = str(count) lhs = f[0].strip() # evaluate bitshifts in constants e.g. "UC_X86 = 1 << 1" match = re.match(r'(?P\s*\d+\s*<<\s*\d+\s*)', rhs) if match: rhs = str(eval(match.group(1))) else: # evaluate references to other constants e.g. "UC_ARM_REG_X = UC_ARM_REG_SP" match = re.match(r'^([^\d]\w+)$', rhs) if match: rhs = previous[match.group(1)] if not rhs.isdigit(): for k, v in previous.items(): rhs = re.sub(r'\b%s\b' % k, v, rhs) rhs = str(eval(rhs)) lhs_strip = re.sub(r'^UC_', '', lhs) count = int(rhs) + 1 if (count == 1): outfile.write(("\n").encode("utf-8")) outfile.write((templ['line_format'] % (lhs_strip, rhs)).encode("utf-8")) previous[lhs] = str(rhs) outfile.write((templ['footer']).encode("utf-8")) outfile.close() def main(): lang = sys.argv[1] if lang == "all": for lang in template.keys(): print("Generating constants for {}".format(lang)) gen(lang) else: if not lang in template: raise RuntimeError("Unsupported binding %s" % lang) gen(lang) if __name__ == "__main__": if len(sys.argv) < 2: print("Usage:", sys.argv[0], " ") print("Supported: {}".format(["all"] + [x for x in template.keys()])) sys.exit(1) main()