blob: ea37c04d4c56ca36ca420c76c64d206d68064cf3 [file] [log] [blame]
Benjamin Peterson90f5ba52010-03-11 22:53:45 +00001#! /usr/bin/env python3
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
R David Murray44b548d2016-09-08 13:59:53 -040026p_define = re.compile(r'^[\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(
R David Murray44b548d2016-09-08 13:59:53 -040029 r'^[\t ]*#[\t ]*define[\t ]+'
30 r'([a-zA-Z0-9_]+)\(([_a-zA-Z][_a-zA-Z0-9]*)\)[\t ]+')
Guido van Rossum07c96451994-10-03 16:45:35 +000031
R David Murray44b548d2016-09-08 13:59:53 -040032p_include = re.compile(r'^[\t ]*#[\t ]*include[\t ]+<([^>\n]+)>')
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
Martin v. Löwisdab3bc02002-11-23 12:08:10 +000041p_hex = re.compile(r"0x([0-9a-fA-F]+)L?")
42
Guido van Rossum07c96451994-10-03 16:45:35 +000043filedict = {}
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000044importable = {}
Guido van Rossum07c96451994-10-03 16:45:35 +000045
Guido van Rossum514d3511995-01-17 17:01:40 +000046try:
Guido van Rossume51c3f52001-12-06 03:24:30 +000047 searchdirs=os.environ['include'].split(';')
Guido van Rossum514d3511995-01-17 17:01:40 +000048except KeyError:
Tim Peters70c43782001-01-17 08:48:39 +000049 try:
Guido van Rossume51c3f52001-12-06 03:24:30 +000050 searchdirs=os.environ['INCLUDE'].split(';')
Tim Peters70c43782001-01-17 08:48:39 +000051 except KeyError:
Antoine Pitrou6103ab12009-10-24 20:11:21 +000052 searchdirs=['/usr/include']
doko@python.org3e6e2ac2013-01-25 13:12:29 +010053 try:
54 searchdirs.insert(0, os.path.join('/usr/include',
55 os.environ['MULTIARCH']))
56 except KeyError:
57 pass
Guido van Rossum514d3511995-01-17 17:01:40 +000058
Guido van Rossum2ba9f301992-03-02 16:20:32 +000059def main():
Tim Peters70c43782001-01-17 08:48:39 +000060 global filedict
61 opts, args = getopt.getopt(sys.argv[1:], 'i:')
62 for o, a in opts:
63 if o == '-i':
Martin v. Löwis4f85bf32001-08-09 12:24:38 +000064 ignores.append(re.compile(a))
Tim Peters70c43782001-01-17 08:48:39 +000065 if not args:
66 args = ['-']
67 for filename in args:
68 if filename == '-':
69 sys.stdout.write('# Generated by h2py from stdin\n')
70 process(sys.stdin, sys.stdout)
71 else:
Serhiy Storchaka172bb392019-03-30 08:33:02 +020072 with open(filename) as fp:
73 outfile = os.path.basename(filename)
74 i = outfile.rfind('.')
75 if i > 0: outfile = outfile[:i]
76 modname = outfile.upper()
77 outfile = modname + '.py'
78 with open(outfile, 'w') as outfp:
79 outfp.write('# Generated by h2py from %s\n' % filename)
80 filedict = {}
81 for dir in searchdirs:
82 if filename[:len(dir)] == dir:
83 filedict[filename[len(dir)+1:]] = None # no '/' trailing
84 importable[filename[len(dir)+1:]] = modname
85 break
86 process(fp, outfp)
Guido van Rossum2ba9f301992-03-02 16:20:32 +000087
Martin v. Löwisdab3bc02002-11-23 12:08:10 +000088def pytify(body):
89 # replace ignored patterns by spaces
90 for p in ignores:
91 body = p.sub(' ', body)
92 # replace char literals by ord(...)
Georg Brandl9c14f2f2010-10-21 13:29:10 +000093 body = p_char.sub("ord('\\1')", body)
Martin v. Löwisdab3bc02002-11-23 12:08:10 +000094 # Compute negative hexadecimal constants
95 start = 0
Christian Heimesa37d4c62007-12-04 23:02:19 +000096 UMAX = 2*(sys.maxsize+1)
Martin v. Löwisdab3bc02002-11-23 12:08:10 +000097 while 1:
98 m = p_hex.search(body, start)
99 if not m: break
100 s,e = m.span()
Georg Brandl8efadf52008-05-16 15:23:30 +0000101 val = int(body[slice(*m.span(1))], 16)
Christian Heimesa37d4c62007-12-04 23:02:19 +0000102 if val > sys.maxsize:
Martin v. Löwisdab3bc02002-11-23 12:08:10 +0000103 val -= UMAX
104 body = body[:s] + "(" + str(val) + ")" + body[e:]
105 start = s + 1
106 return body
107
Guido van Rossum07c96451994-10-03 16:45:35 +0000108def process(fp, outfp, env = {}):
Tim Peters70c43782001-01-17 08:48:39 +0000109 lineno = 0
110 while 1:
111 line = fp.readline()
112 if not line: break
113 lineno = lineno + 1
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000114 match = p_define.match(line)
115 if match:
Tim Peters70c43782001-01-17 08:48:39 +0000116 # gobble up continuation lines
117 while line[-2:] == '\\\n':
118 nextline = fp.readline()
119 if not nextline: break
120 lineno = lineno + 1
121 line = line + nextline
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000122 name = match.group(1)
123 body = line[match.end():]
Martin v. Löwisdab3bc02002-11-23 12:08:10 +0000124 body = pytify(body)
Tim Peters70c43782001-01-17 08:48:39 +0000125 ok = 0
Martin v. Löwisdab3bc02002-11-23 12:08:10 +0000126 stmt = '%s = %s\n' % (name, body.strip())
Tim Peters70c43782001-01-17 08:48:39 +0000127 try:
Georg Brandl7cae87c2006-09-06 06:51:57 +0000128 exec(stmt, env)
Tim Peters70c43782001-01-17 08:48:39 +0000129 except:
130 sys.stderr.write('Skipping: %s' % stmt)
131 else:
132 outfp.write(stmt)
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000133 match = p_macro.match(line)
134 if match:
135 macro, arg = match.group(1, 2)
136 body = line[match.end():]
Martin v. Löwisdab3bc02002-11-23 12:08:10 +0000137 body = pytify(body)
Tim Peters70c43782001-01-17 08:48:39 +0000138 stmt = 'def %s(%s): return %s\n' % (macro, arg, body)
139 try:
Georg Brandl7cae87c2006-09-06 06:51:57 +0000140 exec(stmt, env)
Tim Peters70c43782001-01-17 08:48:39 +0000141 except:
142 sys.stderr.write('Skipping: %s' % stmt)
143 else:
144 outfp.write(stmt)
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000145 match = p_include.match(line)
146 if match:
147 regs = match.regs
Tim Peters70c43782001-01-17 08:48:39 +0000148 a, b = regs[1]
149 filename = line[a:b]
Georg Brandl8efadf52008-05-16 15:23:30 +0000150 if filename in importable:
Martin v. Löwisf2f8c512001-08-09 12:32:10 +0000151 outfp.write('from %s import *\n' % importable[filename])
Georg Brandl8efadf52008-05-16 15:23:30 +0000152 elif filename not in filedict:
Tim Peters70c43782001-01-17 08:48:39 +0000153 filedict[filename] = None
154 inclfp = None
155 for dir in searchdirs:
156 try:
Martin v. Löwis4f85bf32001-08-09 12:24:38 +0000157 inclfp = open(dir + '/' + filename)
Tim Peters70c43782001-01-17 08:48:39 +0000158 break
159 except IOError:
160 pass
161 if inclfp:
Serhiy Storchaka172bb392019-03-30 08:33:02 +0200162 with inclfp:
163 outfp.write(
164 '\n# Included from %s\n' % filename)
165 process(inclfp, outfp, env)
Tim Peters70c43782001-01-17 08:48:39 +0000166 else:
Guido van Rossum436fd752001-12-06 03:28:17 +0000167 sys.stderr.write('Warning - could not find file %s\n' %
168 filename)
Guido van Rossum07c96451994-10-03 16:45:35 +0000169
Andrew M. Kuchlinge236b382004-08-09 17:27:55 +0000170if __name__ == '__main__':
171 main()