blob: 55f99546e2b9c4f1596b4f30b1f4a0069d4e4706 [file] [log] [blame]
Martin v. Löwis497114e2006-03-01 05:18:07 +00001#! /usr/bin/env python
2from __future__ import with_statement
3import errno
4import os
5import re
6import sys
7import string
8
9if __name__ == "__main__":
10 _base = sys.argv[0]
11else:
12 _base = __file__
13
14_script_home = os.path.abspath(os.path.dirname(_base))
15
16srcdir = os.path.dirname(os.path.dirname(_script_home))
17
18EXCLUDES = ["bitset.h", "cStringIO.h", "graminit.h", "grammar.h",
19 "longintrepr.h", "metagrammar.h",
20 "node.h", "opcode.h", "osdefs.h", "pgenheaders.h",
21 "py_curses.h", "parsetok.h", "symtable.h", "token.h"]
22
23
24def list_headers():
25 """Return a list of headers."""
26 incdir = os.path.join(srcdir, "Include")
27 return [os.path.join(incdir, fn) for fn in os.listdir(incdir)
28 if fn.endswith(".h") and fn not in EXCLUDES]
29
30
31def matcher(pattern):
32 return re.compile(pattern).search
33
34MATCHERS = [
35 # XXX this should also deal with ctypedesc, cvardesc and cmemberdesc
36 matcher(r"\\begin\{cfuncdesc\}\{(?P<result>[^}]*)\}\{(?P<sym>[^}]*)\}{(?P<params>[^}]*)\}"),
37 matcher(r"\\cfuncline\{(?P<result>[^})]*)\}\{(?P<sym>[^}]*)\}{(?P<params>[^}]*)\}"),
38 ]
39
40def list_documented_items():
41 """Return a list of everything that's already documented."""
42 apidir = os.path.join(srcdir, "Doc", "api")
43 files = [fn for fn in os.listdir(apidir) if fn.endswith(".tex")]
44 L = []
45 for fn in files:
46 fullname = os.path.join(apidir, fn)
47 data = open(fullname).read()
48 for matcher in MATCHERS:
49 pos = 0
50 while 1:
51 m = matcher(data, pos)
52 if not m: break
53 pos = m.end()
54 sym = m.group("sym")
55 result = m.group("result")
56 params = m.group("params")
57 # replace all whitespace with a single one
58 params = " ".join(params.split())
59 L.append((sym, result, params, fn))
60 return L
61
62def normalize_type(t):
63 t = t.strip()
64 s = t.rfind("*")
65 if s != -1:
66 # strip everything after the pointer name
67 t = t[:s+1]
68 # Drop the variable name
69 s = t.split()
70 typenames = 1
71 if len(s)>1 and s[0]=='unsigned' and s[1]=='int':
72 typenames = 2
73 if len(s) > typenames and s[-1][0] in string.letters:
74 del s[-1]
75 if not s:
76 print "XXX", t
77 return ""
78 # Drop register
79 if s[0] == "register":
80 del s[0]
81 # discard all spaces
82 return ''.join(s)
83
84def compare_type(t1, t2):
85 t1 = normalize_type(t1)
86 t2 = normalize_type(t2)
87 if t1 == r'\moreargs' and t2 == '...':
88 return False
89 if t1 != t2:
90 #print "different:", t1, t2
91 return False
92 return True
93
94
95def compare_types(ret, params, hret, hparams):
96 if not compare_type(ret, hret):
97 return False
98 params = params.split(",")
99 hparams = hparams.split(",")
100 if not params and hparams == ['void']:
101 return True
102 if not hparams and params == ['void']:
103 return True
104 if len(params) != len(hparams):
105 return False
106 for p1, p2 in zip(params, hparams):
107 if not compare_type(p1, p2):
108 return False
109 return True
110
111def main():
112 headers = list_headers()
113 documented = list_documented_items()
114
115 lines = []
116 for h in headers:
117 data = open(h).read()
118 data, n = re.subn(r"PyAPI_FUNC\(([^)]*)\)", r"\1", data)
119 name = os.path.basename(h)
120 with open(name, "w") as f:
121 f.write(data)
122 cmd = ("ctags -f - --file-scope=no --c-kinds=p --fields=S "
123 "-Istaticforward -Istatichere=static " + name)
124 with os.popen(cmd) as f:
125 lines.extend(f.readlines())
126 os.unlink(name)
127 L = {}
128 prevsym = None
129 for line in lines:
130 if not line:
131 break
132 sym, filename, signature = line.split(None, 2)
133 if sym == prevsym:
134 continue
135 expr = "\^(.*)%s" % sym
136 m = re.search(expr, signature)
137 if not m:
138 print "Could not split",signature, "using",expr
139 rettype = m.group(1).strip()
140 m = re.search("signature:\(([^)]*)\)", signature)
141 if not m:
142 print "Could not get signature from", signature
143 params = m.group(1)
144 L[sym] = (rettype, params)
145
146 for sym, ret, params, fn in documented:
147 if sym not in L:
148 print "No declaration for '%s'" % sym
149 continue
150 hret, hparams = L[sym]
151 if not compare_types(ret, params, hret, hparams):
152 print "Declaration error for %s (%s):" % (sym, fn)
153 print ret+": "+params
154 print hret+": "+hparams
155
156if __name__ == "__main__":
157 main()