Guido van Rossum | f06ee5f | 1996-11-27 19:52:01 +0000 | [diff] [blame] | 1 | #! /usr/bin/env python |
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: |
| 52 | try: |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 53 | if sys.platform.find("beos") == 0: |
Guido van Rossum | e51c3f5 | 2001-12-06 03:24:30 +0000 | [diff] [blame] | 54 | searchdirs=os.environ['BEINCLUDES'].split(';') |
Martin v. Löwis | f90ae20 | 2002-06-11 06:22:31 +0000 | [diff] [blame] | 55 | elif sys.platform.startswith("atheos"): |
| 56 | searchdirs=os.environ['C_INCLUDE_PATH'].split(':') |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 57 | else: |
| 58 | raise KeyError |
| 59 | except KeyError: |
| 60 | searchdirs=['/usr/include'] |
Guido van Rossum | 514d351 | 1995-01-17 17:01:40 +0000 | [diff] [blame] | 61 | |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 62 | def main(): |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 63 | global filedict |
| 64 | opts, args = getopt.getopt(sys.argv[1:], 'i:') |
| 65 | for o, a in opts: |
| 66 | if o == '-i': |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 67 | ignores.append(re.compile(a)) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 68 | if not args: |
| 69 | args = ['-'] |
| 70 | for filename in args: |
| 71 | if filename == '-': |
| 72 | sys.stdout.write('# Generated by h2py from stdin\n') |
| 73 | process(sys.stdin, sys.stdout) |
| 74 | else: |
| 75 | fp = open(filename, 'r') |
| 76 | outfile = os.path.basename(filename) |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 77 | i = outfile.rfind('.') |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 78 | if i > 0: outfile = outfile[:i] |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 79 | modname = outfile.upper() |
| 80 | outfile = modname + '.py' |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 81 | outfp = open(outfile, 'w') |
| 82 | outfp.write('# Generated by h2py from %s\n' % filename) |
| 83 | filedict = {} |
| 84 | for dir in searchdirs: |
| 85 | if filename[:len(dir)] == dir: |
| 86 | filedict[filename[len(dir)+1:]] = None # no '/' trailing |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 87 | importable[filename[len(dir)+1:]] = modname |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 88 | break |
| 89 | process(fp, outfp) |
| 90 | outfp.close() |
| 91 | fp.close() |
Guido van Rossum | 2ba9f30 | 1992-03-02 16:20:32 +0000 | [diff] [blame] | 92 | |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 93 | def pytify(body): |
| 94 | # replace ignored patterns by spaces |
| 95 | for p in ignores: |
| 96 | body = p.sub(' ', body) |
| 97 | # replace char literals by ord(...) |
| 98 | body = p_char.sub('ord(\\0)', body) |
| 99 | # Compute negative hexadecimal constants |
| 100 | start = 0 |
| 101 | UMAX = 2*(sys.maxint+1) |
| 102 | while 1: |
| 103 | m = p_hex.search(body, start) |
| 104 | if not m: break |
| 105 | s,e = m.span() |
| 106 | val = long(body[slice(*m.span(1))], 16) |
| 107 | if val > sys.maxint: |
| 108 | val -= UMAX |
| 109 | body = body[:s] + "(" + str(val) + ")" + body[e:] |
| 110 | start = s + 1 |
| 111 | return body |
| 112 | |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 113 | def process(fp, outfp, env = {}): |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 114 | lineno = 0 |
| 115 | while 1: |
| 116 | line = fp.readline() |
| 117 | if not line: break |
| 118 | lineno = lineno + 1 |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 119 | match = p_define.match(line) |
| 120 | if match: |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 121 | # gobble up continuation lines |
| 122 | while line[-2:] == '\\\n': |
| 123 | nextline = fp.readline() |
| 124 | if not nextline: break |
| 125 | lineno = lineno + 1 |
| 126 | line = line + nextline |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 127 | name = match.group(1) |
| 128 | body = line[match.end():] |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 129 | body = pytify(body) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 130 | ok = 0 |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 131 | stmt = '%s = %s\n' % (name, body.strip()) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 132 | try: |
Georg Brandl | 7cae87c | 2006-09-06 06:51:57 +0000 | [diff] [blame] | 133 | exec(stmt, env) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 134 | except: |
| 135 | sys.stderr.write('Skipping: %s' % stmt) |
| 136 | else: |
| 137 | outfp.write(stmt) |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 138 | match = p_macro.match(line) |
| 139 | if match: |
| 140 | macro, arg = match.group(1, 2) |
| 141 | body = line[match.end():] |
Martin v. Löwis | dab3bc0 | 2002-11-23 12:08:10 +0000 | [diff] [blame] | 142 | body = pytify(body) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 143 | stmt = 'def %s(%s): return %s\n' % (macro, arg, body) |
| 144 | try: |
Georg Brandl | 7cae87c | 2006-09-06 06:51:57 +0000 | [diff] [blame] | 145 | exec(stmt, env) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 146 | except: |
| 147 | sys.stderr.write('Skipping: %s' % stmt) |
| 148 | else: |
| 149 | outfp.write(stmt) |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 150 | match = p_include.match(line) |
| 151 | if match: |
| 152 | regs = match.regs |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 153 | a, b = regs[1] |
| 154 | filename = line[a:b] |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 155 | if importable.has_key(filename): |
Martin v. Löwis | f2f8c51 | 2001-08-09 12:32:10 +0000 | [diff] [blame] | 156 | outfp.write('from %s import *\n' % importable[filename]) |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 157 | elif not filedict.has_key(filename): |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 158 | filedict[filename] = None |
| 159 | inclfp = None |
| 160 | for dir in searchdirs: |
| 161 | try: |
Martin v. Löwis | 4f85bf3 | 2001-08-09 12:24:38 +0000 | [diff] [blame] | 162 | inclfp = open(dir + '/' + filename) |
Tim Peters | 70c4378 | 2001-01-17 08:48:39 +0000 | [diff] [blame] | 163 | break |
| 164 | except IOError: |
| 165 | pass |
| 166 | if inclfp: |
| 167 | outfp.write( |
| 168 | '\n# Included from %s\n' % filename) |
| 169 | process(inclfp, outfp, env) |
| 170 | else: |
Guido van Rossum | 436fd75 | 2001-12-06 03:28:17 +0000 | [diff] [blame] | 171 | sys.stderr.write('Warning - could not find file %s\n' % |
| 172 | filename) |
Guido van Rossum | 07c9645 | 1994-10-03 16:45:35 +0000 | [diff] [blame] | 173 | |
Andrew M. Kuchling | e236b38 | 2004-08-09 17:27:55 +0000 | [diff] [blame] | 174 | if __name__ == '__main__': |
| 175 | main() |