Benjamin Peterson | 90f5ba5 | 2010-03-11 22:53:45 +0000 | [diff] [blame] | 1 | #! /usr/bin/env python3 |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 2 | |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 3 | # Read #define's and translate to Python code. |
| 4 | # Handle #include statements. |
| 5 | # Handle #define macros with one argument. |
| 6 | # Anything that isn't recognized or doesn't translate into valid |
| 7 | # Python is ignored. |
| 8 | |
| 9 | # Without filename arguments, acts as a filter. |
Guido van Rossum | 09336f9 | 1994-05-03 14:37:30 +0000 | [diff] [blame] | 10 | # If one or more filenames are given, output is written to corresponding |
| 11 | # filenames in the local directory, translated to all uppercase, with |
| 12 | # the extension replaced by ".py". |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 13 | |
Guido van Rossum | 01f5f62 | 1994-05-17 09:05:54 +0000 | [diff] [blame] | 14 | # By passing one or more options of the form "-i regular_expression" |
| 15 | # you can specify additional strings to be ignored. This is useful |
| 16 | # e.g. to ignore casts to u_long: simply specify "-i '(u_long)'". |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 17 | |
| 18 | # XXX To do: |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 19 | # - turn trailing C comments into Python comments |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 20 | # - turn C Boolean operators "&& || !" into Python "and or not" |
| 21 | # - what to do about #if(def)? |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 22 | # - what to do about macros with multiple parameters? |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 23 | |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 24 | import sys, re, getopt, os |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 25 | |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 26 | p_define = re.compile('^[\t ]*#[\t ]*define[\t ]+([a-zA-Z0-9_]+)[\t ]+') |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 27 | |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 28 | p_macro = re.compile( |
Guido van Rossum | 6100d91 | 1996-08-22 23:12:23 +0000 | [diff] [blame] | 29 | '^[\t ]*#[\t ]*define[\t ]+' |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 30 | '([a-zA-Z0-9_]+)\(([_a-zA-Z][_a-zA-Z0-9]*)\)[\t ]+') |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 31 | |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 32 | p_include = re.compile('^[\t ]*#[\t ]*include[\t ]+<([a-zA-Z0-9_/\.]+)') |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 33 | |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 34 | p_comment = re.compile(r'/\*([^*]+|\*+[^/])*(\*+/)?') |
| 35 | p_cpp_comment = re.compile('//.*') |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 36 | |
Guido van Rossum | 9189bda | 1997-08-14 20:14:29 +0000 | [diff] [blame] | 37 | ignores = [p_comment, p_cpp_comment] |
Guido van Rossum | 01f5f62 | 1994-05-17 09:05:54 +0000 | [diff] [blame] | 38 | |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 39 | p_char = re.compile(r"'(\\.[^\\]*|[^\\])'") |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 40 | |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 41 | p_hex = re.compile(r"0x([0-9a-fA-F]+)L?") |
| 42 | |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 43 | filedict = {} |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 44 | importable = {} |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 45 | |
Guido van Rossum | 514d351 | 1995-01-17 17:01:40 +0000 | [diff] [blame] | 46 | try: |
Guido van Rossum | e51c3f5 | 2001-12-06 03:24:30 +0000 | [diff] [blame] | 47 | searchdirs=os.environ['include'].split(';') |
Guido van Rossum | 514d351 | 1995-01-17 17:01:40 +0000 | [diff] [blame] | 48 | except KeyError: |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 49 | try: |
Guido van Rossum | e51c3f5 | 2001-12-06 03:24:30 +0000 | [diff] [blame] | 50 | searchdirs=os.environ['INCLUDE'].split(';') |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 51 | except KeyError: |
Antoine Pitrou | 6103ab1 | 2009-10-24 20:11:21 +0000 | [diff] [blame] | 52 | searchdirs=['/usr/include'] |
Guido van Rossum | 514d351 | 1995-01-17 17:01:40 +0000 | [diff] [blame] | 53 | |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 54 | def main(): |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 55 | global filedict |
| 56 | opts, args = getopt.getopt(sys.argv[1:], 'i:') |
| 57 | for o, a in opts: |
| 58 | if o == '-i': |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 59 | ignores.append(re.compile(a)) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 60 | if not args: |
| 61 | args = ['-'] |
| 62 | for filename in args: |
| 63 | if filename == '-': |
| 64 | sys.stdout.write('# Generated by h2py from stdin\n') |
| 65 | process(sys.stdin, sys.stdout) |
| 66 | else: |
| 67 | fp = open(filename, 'r') |
| 68 | outfile = os.path.basename(filename) |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 69 | i = outfile.rfind('.') |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 70 | if i > 0: outfile = outfile[:i] |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 71 | modname = outfile.upper() |
| 72 | outfile = modname + '.py' |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 73 | outfp = open(outfile, 'w') |
| 74 | outfp.write('# Generated by h2py from %s\n' % filename) |
| 75 | filedict = {} |
| 76 | for dir in searchdirs: |
| 77 | if filename[:len(dir)] == dir: |
| 78 | filedict[filename[len(dir)+1:]] = None # no '/' trailing |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 79 | importable[filename[len(dir)+1:]] = modname |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 80 | break |
| 81 | process(fp, outfp) |
| 82 | outfp.close() |
| 83 | fp.close() |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 84 | |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 85 | def pytify(body): |
| 86 | # replace ignored patterns by spaces |
| 87 | for p in ignores: |
| 88 | body = p.sub(' ', body) |
| 89 | # replace char literals by ord(...) |
Georg Brandl | 9c14f2f | 2010-10-21 13:29:10 +0000 | [diff] [blame] | 90 | body = p_char.sub("ord('\\1')", body) |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 91 | # Compute negative hexadecimal constants |
| 92 | start = 0 |
Christian Heimes | a37d4c6 | 2007-12-04 23:02:19 +0000 | [diff] [blame] | 93 | UMAX = 2*(sys.maxsize+1) |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 94 | while 1: |
| 95 | m = p_hex.search(body, start) |
| 96 | if not m: break |
| 97 | s,e = m.span() |
Georg Brandl | 8efadf5 | 2008-05-16 15:23:30 +0000 | [diff] [blame] | 98 | val = int(body[slice(*m.span(1))], 16) |
Christian Heimes | a37d4c6 | 2007-12-04 23:02:19 +0000 | [diff] [blame] | 99 | if val > sys.maxsize: |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 100 | val -= UMAX |
| 101 | body = body[:s] + "(" + str(val) + ")" + body[e:] |
| 102 | start = s + 1 |
| 103 | return body |
| 104 | |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 105 | def process(fp, outfp, env = {}): |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 106 | lineno = 0 |
| 107 | while 1: |
| 108 | line = fp.readline() |
| 109 | if not line: break |
| 110 | lineno = lineno + 1 |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 111 | match = p_define.match(line) |
| 112 | if match: |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 113 | # gobble up continuation lines |
| 114 | while line[-2:] == '\\\n': |
| 115 | nextline = fp.readline() |
| 116 | if not nextline: break |
| 117 | lineno = lineno + 1 |
| 118 | line = line + nextline |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 119 | name = match.group(1) |
| 120 | body = line[match.end():] |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 121 | body = pytify(body) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 122 | ok = 0 |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 123 | stmt = '%s = %s\n' % (name, body.strip()) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 124 | try: |
Georg Brandl | 7cae87c | 2006-09-06 06:51:57 +0000 | [diff] [blame] | 125 | exec(stmt, env) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 126 | except: |
| 127 | sys.stderr.write('Skipping: %s' % stmt) |
| 128 | else: |
| 129 | outfp.write(stmt) |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 130 | match = p_macro.match(line) |
| 131 | if match: |
| 132 | macro, arg = match.group(1, 2) |
| 133 | body = line[match.end():] |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 134 | body = pytify(body) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 135 | stmt = 'def %s(%s): return %s\n' % (macro, arg, body) |
| 136 | try: |
Georg Brandl | 7cae87c | 2006-09-06 06:51:57 +0000 | [diff] [blame] | 137 | exec(stmt, env) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 138 | except: |
| 139 | sys.stderr.write('Skipping: %s' % stmt) |
| 140 | else: |
| 141 | outfp.write(stmt) |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 142 | match = p_include.match(line) |
| 143 | if match: |
| 144 | regs = match.regs |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 145 | a, b = regs[1] |
| 146 | filename = line[a:b] |
Georg Brandl | 8efadf5 | 2008-05-16 15:23:30 +0000 | [diff] [blame] | 147 | if filename in importable: |
Martin v. Löwis | f2f8c51 | 2001-08-09 12:32:10 +0000 | [diff] [blame] | 148 | outfp.write('from %s import *\n' % importable[filename]) |
Georg Brandl | 8efadf5 | 2008-05-16 15:23:30 +0000 | [diff] [blame] | 149 | elif filename not in filedict: |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 150 | filedict[filename] = None |
| 151 | inclfp = None |
| 152 | for dir in searchdirs: |
| 153 | try: |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 154 | inclfp = open(dir + '/' + filename) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 155 | break |
| 156 | except IOError: |
| 157 | pass |
| 158 | if inclfp: |
| 159 | outfp.write( |
| 160 | '\n# Included from %s\n' % filename) |
| 161 | process(inclfp, outfp, env) |
| 162 | else: |
Guido van Rossum | 436fd75 | 2001-12-06 03:28:17 +0000 | [diff] [blame] | 163 | sys.stderr.write('Warning - could not find file %s\n' % |
| 164 | filename) |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 165 | |
Andrew M. Kuchling | e236b38 | 2004-08-09 17:27:55 +0000 | [diff] [blame] | 166 | if __name__ == '__main__': |
| 167 | main() |