blob: 4e81c865d87a751eb2382e465014eae49adfd6cb [file] [log] [blame]
Guido van Rossumf06ee5f1996-11-27 19:52:01 +00001#! /usr/bin/env python
Guido van Rossum2ba9f301992-03-02 16:20:32 +00002
Guido van Rossum07c96451994-10-03 16:45:35 +00003# 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 Rossum09336f91994-05-03 14:37:30 +000010# 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 Rossum07c96451994-10-03 16:45:35 +000013
Guido van Rossum01f5f621994-05-17 09:05:54 +000014# 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 Rossum2ba9f301992-03-02 16:20:32 +000017
18# XXX To do:
Guido van Rossum2ba9f301992-03-02 16:20:32 +000019# - turn trailing C comments into Python comments
Guido van Rossum2ba9f301992-03-02 16:20:32 +000020# - turn C Boolean operators "&& || !" into Python "and or not"
21# - what to do about #if(def)?
Guido van Rossum07c96451994-10-03 16:45:35 +000022# - what to do about macros with multiple parameters?
Guido van Rossum2ba9f301992-03-02 16:20:32 +000023
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000024import sys, re, getopt, os
Guido van Rossum2ba9f301992-03-02 16:20:32 +000025
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000026p_define = re.compile('^[\t ]*#[\t ]*define[\t ]+([a-zA-Z0-9_]+)[\t ]+')
Guido van Rossum2ba9f301992-03-02 16:20:32 +000027
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000028p_macro = re.compile(
Guido van Rossum6100d911996-08-22 23:12:23 +000029 '^[\t ]*#[\t ]*define[\t ]+'
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000030 '([a-zA-Z0-9_]+)\(([_a-zA-Z][_a-zA-Z0-9]*)\)[\t ]+')
Guido van Rossum07c96451994-10-03 16:45:35 +000031
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000032p_include = re.compile('^[\t ]*#[\t ]*include[\t ]+<([a-zA-Z0-9_/\.]+)')
Guido van Rossum07c96451994-10-03 16:45:35 +000033
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000034p_comment = re.compile(r'/\*([^*]+|\*+[^/])*(\*+/)?')
35p_cpp_comment = re.compile('//.*')
Guido van Rossum2ba9f301992-03-02 16:20:32 +000036
Guido van Rossum9189bda1997-08-14 20:14:29 +000037ignores = [p_comment, p_cpp_comment]
Guido van Rossum01f5f621994-05-17 09:05:54 +000038
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000039p_char = re.compile(r"'(\\.[^\\]*|[^\\])'")
Guido van Rossum07c96451994-10-03 16:45:35 +000040
41filedict = {}
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000042importable = {}
Guido van Rossum07c96451994-10-03 16:45:35 +000043
Guido van Rossum514d3511995-01-17 17:01:40 +000044try:
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000045 searchdirs=os.environ['include'].splitfields(';')
Guido van Rossum514d3511995-01-17 17:01:40 +000046except KeyError:
Tim Peters70c43782001-01-17 08:48:39 +000047 try:
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000048 searchdirs=os.environ['INCLUDE'].splitfields(';')
Tim Peters70c43782001-01-17 08:48:39 +000049 except KeyError:
50 try:
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000051 if sys.platform.find("beos") == 0:
52 searchdirs=os.environ['BEINCLUDES'].splitfields(';')
Tim Peters70c43782001-01-17 08:48:39 +000053 else:
54 raise KeyError
55 except KeyError:
56 searchdirs=['/usr/include']
Guido van Rossum514d3511995-01-17 17:01:40 +000057
Guido van Rossum2ba9f301992-03-02 16:20:32 +000058def main():
Tim Peters70c43782001-01-17 08:48:39 +000059 global filedict
60 opts, args = getopt.getopt(sys.argv[1:], 'i:')
61 for o, a in opts:
62 if o == '-i':
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000063 ignores.append(re.compile(a))
Tim Peters70c43782001-01-17 08:48:39 +000064 if not args:
65 args = ['-']
66 for filename in args:
67 if filename == '-':
68 sys.stdout.write('# Generated by h2py from stdin\n')
69 process(sys.stdin, sys.stdout)
70 else:
71 fp = open(filename, 'r')
72 outfile = os.path.basename(filename)
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000073 i = outfile.rfind('.')
Tim Peters70c43782001-01-17 08:48:39 +000074 if i > 0: outfile = outfile[:i]
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000075 modname = outfile.upper()
76 outfile = modname + '.py'
Tim Peters70c43782001-01-17 08:48:39 +000077 outfp = open(outfile, 'w')
78 outfp.write('# Generated by h2py from %s\n' % filename)
79 filedict = {}
80 for dir in searchdirs:
81 if filename[:len(dir)] == dir:
82 filedict[filename[len(dir)+1:]] = None # no '/' trailing
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000083 importable[filename[len(dir)+1:]] = modname
Tim Peters70c43782001-01-17 08:48:39 +000084 break
85 process(fp, outfp)
86 outfp.close()
87 fp.close()
Guido van Rossum2ba9f301992-03-02 16:20:32 +000088
Guido van Rossum07c96451994-10-03 16:45:35 +000089def process(fp, outfp, env = {}):
Tim Peters70c43782001-01-17 08:48:39 +000090 lineno = 0
91 while 1:
92 line = fp.readline()
93 if not line: break
94 lineno = lineno + 1
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000095 match = p_define.match(line)
96 if match:
Tim Peters70c43782001-01-17 08:48:39 +000097 # gobble up continuation lines
98 while line[-2:] == '\\\n':
99 nextline = fp.readline()
100 if not nextline: break
101 lineno = lineno + 1
102 line = line + nextline
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000103 name = match.group(1)
104 body = line[match.end():]
Tim Peters70c43782001-01-17 08:48:39 +0000105 # replace ignored patterns by spaces
106 for p in ignores:
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000107 body = p.sub(' ', body)
Tim Peters70c43782001-01-17 08:48:39 +0000108 # replace char literals by ord(...)
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000109 body = p_char.sub('ord(\\0)', body)
110 stmt = '%s = %s\n' % (name, body.strip())
Tim Peters70c43782001-01-17 08:48:39 +0000111 ok = 0
112 try:
113 exec stmt in env
114 except:
115 sys.stderr.write('Skipping: %s' % stmt)
116 else:
117 outfp.write(stmt)
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000118 match = p_macro.match(line)
119 if match:
120 macro, arg = match.group(1, 2)
121 body = line[match.end():]
Tim Peters70c43782001-01-17 08:48:39 +0000122 for p in ignores:
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000123 body = p.sub(' ', body)
124 body = p_char.sub('ord(\\0)', body)
Tim Peters70c43782001-01-17 08:48:39 +0000125 stmt = 'def %s(%s): return %s\n' % (macro, arg, body)
126 try:
127 exec stmt in env
128 except:
129 sys.stderr.write('Skipping: %s' % stmt)
130 else:
131 outfp.write(stmt)
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000132 match = p_include.match(line)
133 if match:
134 regs = match.regs
Tim Peters70c43782001-01-17 08:48:39 +0000135 a, b = regs[1]
136 filename = line[a:b]
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000137 if importable.has_key(filename):
138 outfp.write('import %s\n' % importable[filename])
139 elif not filedict.has_key(filename):
Tim Peters70c43782001-01-17 08:48:39 +0000140 filedict[filename] = None
141 inclfp = None
142 for dir in searchdirs:
143 try:
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000144 inclfp = open(dir + '/' + filename)
Tim Peters70c43782001-01-17 08:48:39 +0000145 break
146 except IOError:
147 pass
148 if inclfp:
149 outfp.write(
150 '\n# Included from %s\n' % filename)
151 process(inclfp, outfp, env)
152 else:
153 sys.stderr.write('Warning - could not find file %s' % filename)
Guido van Rossum07c96451994-10-03 16:45:35 +0000154
Guido van Rossum514d3511995-01-17 17:01:40 +0000155main()