Fix to python API extractor and API doc generation

This fixes a number of issues most of them raised by Eric Blake on the
generated documentation output:
   - parsing of "long long int" and similar
   - add parsing of unions within a struct
   - remove spurious " * " fron comments on structure fields and enums
   - fix concatenation of base type and name in arrays
   - extend XSLT to cope with union in structs

* docs/apibuild.py: fix and extend API extraction tool
* docs/newapi.xsl: extend the stylesheets to cope with union in
  public structures
This commit is contained in:
Daniel Veillard 2011-06-20 11:25:34 +08:00
parent 017abcbb1a
commit d42ea21aec
2 changed files with 206 additions and 33 deletions

View File

@ -179,6 +179,7 @@ class index:
self.variables = {} self.variables = {}
self.includes = {} self.includes = {}
self.structs = {} self.structs = {}
self.unions = {}
self.enums = {} self.enums = {}
self.typedefs = {} self.typedefs = {}
self.macros = {} self.macros = {}
@ -232,6 +233,10 @@ class index:
self.includes[name] = d self.includes[name] = d
elif type == "struct": elif type == "struct":
self.structs[name] = d self.structs[name] = d
elif type == "struct":
self.structs[name] = d
elif type == "union":
self.unions[name] = d
elif type == "enum": elif type == "enum":
self.enums[name] = d self.enums[name] = d
elif type == "typedef": elif type == "typedef":
@ -280,6 +285,13 @@ class index:
else: else:
self.structs[id] = idx.structs[id] self.structs[id] = idx.structs[id]
self.identifiers[id] = idx.structs[id] self.identifiers[id] = idx.structs[id]
for id in idx.unions.keys():
if self.unions.has_key(id):
print "union %s from %s redeclared in %s" % (
id, self.unions[id].header, idx.unions[id].header)
else:
self.unions[id] = idx.unions[id]
self.identifiers[id] = idx.unions[id]
for id in idx.typedefs.keys(): for id in idx.typedefs.keys():
if self.typedefs.has_key(id): if self.typedefs.has_key(id):
print "typedef %s from %s redeclared in %s" % ( print "typedef %s from %s redeclared in %s" % (
@ -347,6 +359,7 @@ class index:
self.analyze_dict("functions", self.functions) self.analyze_dict("functions", self.functions)
self.analyze_dict("variables", self.variables) self.analyze_dict("variables", self.variables)
self.analyze_dict("structs", self.structs) self.analyze_dict("structs", self.structs)
self.analyze_dict("unions", self.unions)
self.analyze_dict("typedefs", self.typedefs) self.analyze_dict("typedefs", self.typedefs)
self.analyze_dict("macros", self.macros) self.analyze_dict("macros", self.macros)
@ -656,13 +669,36 @@ class CParser:
res[item] = line res[item] = line
self.index.info = res self.index.info = res
def strip_lead_star(self, line):
l = len(line)
i = 0
while i < l:
if line[i] == ' ' or line[i] == '\t':
i += 1
elif line[i] == '*':
return line[:i] + line[i + 1:]
else:
return line
return line
def cleanupComment(self):
if type(self.comment) != type(""):
return
# remove the leading * on multi-line comments
lines = self.comment.splitlines(True)
com = ""
for line in lines:
com = com + self.strip_lead_star(line)
self.comment = com.strip()
def parseComment(self, token): def parseComment(self, token):
com = token[1]
if self.top_comment == "": if self.top_comment == "":
self.top_comment = token[1] self.top_comment = com
if self.comment == None or token[1][0] == '*': if self.comment == None or com[0] == '*':
self.comment = token[1]; self.comment = com;
else: else:
self.comment = self.comment + token[1] self.comment = self.comment + com
token = self.lexer.token() token = self.lexer.token()
if string.find(self.comment, "DOC_DISABLE") != -1: if string.find(self.comment, "DOC_DISABLE") != -1:
@ -1178,7 +1214,13 @@ class CParser:
if token[0] == "sep" and token[1] == ";": if token[0] == "sep" and token[1] == ";":
self.comment = None self.comment = None
token = self.token() token = self.token()
fields.append((self.type, fname, self.comment)) self.cleanupComment()
if self.type == "union":
fields.append((self.type, fname, self.comment,
self.union_fields))
self.union_fields = []
else:
fields.append((self.type, fname, self.comment))
self.comment = None self.comment = None
else: else:
self.error("parseStruct: expecting ;", token) self.error("parseStruct: expecting ;", token)
@ -1200,6 +1242,56 @@ class CParser:
#print fields #print fields
return token return token
#
# Parse a C union definition till the balancing }
#
def parseUnion(self, token):
fields = []
# self.debug("start parseUnion", token)
while token != None:
if token[0] == "sep" and token[1] == "{":
token = self.token()
token = self.parseTypeBlock(token)
elif token[0] == "sep" and token[1] == "}":
self.union_fields = fields
# self.debug("end parseUnion", token)
# print fields
token = self.token()
return token
else:
base_type = self.type
# self.debug("before parseType", token)
token = self.parseType(token)
# self.debug("after parseType", token)
if token != None and token[0] == "name":
fname = token[1]
token = self.token()
if token[0] == "sep" and token[1] == ";":
self.comment = None
token = self.token()
self.cleanupComment()
fields.append((self.type, fname, self.comment))
self.comment = None
else:
self.error("parseUnion: expecting ;", token)
elif token != None and token[0] == "sep" and token[1] == "{":
token = self.token()
token = self.parseTypeBlock(token)
if token != None and token[0] == "name":
token = self.token()
if token != None and token[0] == "sep" and token[1] == ";":
token = self.token()
else:
self.error("parseUnion: expecting ;", token)
else:
self.error("parseUnion: name", token)
token = self.token()
self.type = base_type;
self.union_fields = fields
# self.debug("end parseUnion", token)
# print fields
return token
# #
# Parse a C enum block, parse till the balancing } # Parse a C enum block, parse till the balancing }
# #
@ -1215,6 +1307,7 @@ class CParser:
token = self.parseTypeBlock(token) token = self.parseTypeBlock(token)
elif token[0] == "sep" and token[1] == "}": elif token[0] == "sep" and token[1] == "}":
if name != None: if name != None:
self.cleanupComment()
if self.comment != None: if self.comment != None:
comment = self.comment comment = self.comment
self.comment = None self.comment = None
@ -1222,6 +1315,7 @@ class CParser:
token = self.token() token = self.token()
return token return token
elif token[0] == "name": elif token[0] == "name":
self.cleanupComment()
if name != None: if name != None:
if self.comment != None: if self.comment != None:
comment = string.strip(self.comment) comment = string.strip(self.comment)
@ -1252,7 +1346,7 @@ class CParser:
return token return token
# #
# Parse a C definition block, used for structs it parse till # Parse a C definition block, used for structs or unions it parse till
# the balancing } # the balancing }
# #
def parseTypeBlock(self, token): def parseTypeBlock(self, token):
@ -1275,6 +1369,7 @@ class CParser:
def parseType(self, token): def parseType(self, token):
self.type = "" self.type = ""
self.struct_fields = [] self.struct_fields = []
self.union_fields = []
self.signature = None self.signature = None
if token == None: if token == None:
return token return token
@ -1304,22 +1399,19 @@ class CParser:
self.push(token) self.push(token)
token = oldtmp token = oldtmp
oldtmp = token
token = self.token()
if token[0] == "name" and token[1] == "int": if token[0] == "name" and token[1] == "int":
if self.type == "": self.type = self.type + " " + token[1]
self.type = tmp[1] else:
else: self.push(token)
self.type = self.type + " " + tmp[1] token = oldtmp
elif token[0] == "name" and token[1] == "short": elif token[0] == "name" and token[1] == "short":
if self.type == "": if self.type == "":
self.type = token[1] self.type = token[1]
else: else:
self.type = self.type + " " + token[1] self.type = self.type + " " + token[1]
if token[0] == "name" and token[1] == "int":
if self.type == "":
self.type = tmp[1]
else:
self.type = self.type + " " + tmp[1]
elif token[0] == "name" and token[1] == "struct": elif token[0] == "name" and token[1] == "struct":
if self.type == "": if self.type == "":
@ -1355,6 +1447,28 @@ class CParser:
token = nametok token = nametok
return token return token
elif token[0] == "name" and token[1] == "union":
if self.type == "":
self.type = token[1]
else:
self.type = self.type + " " + token[1]
token = self.token()
nametok = None
if token[0] == "name":
nametok = token
token = self.token()
if token != None and token[0] == "sep" and token[1] == "{":
token = self.token()
token = self.parseUnion(token)
elif token != None and token[0] == "name" and nametok != None:
self.type = self.type + " " + nametok[1]
return token
if nametok != None:
self.lexer.push(token)
token = nametok
return token
elif token[0] == "name" and token[1] == "enum": elif token[0] == "name" and token[1] == "enum":
if self.type == "": if self.type == "":
self.type = token[1] self.type = token[1]
@ -1434,7 +1548,7 @@ class CParser:
nametok = token nametok = token
token = self.token() token = self.token()
if token != None and token[0] == "sep" and token[1] == '[': if token != None and token[0] == "sep" and token[1] == '[':
self.type = self.type + nametok[1] self.type = self.type + " " + nametok[1]
while token != None and token[0] == "sep" and token[1] == '[': while token != None and token[0] == "sep" and token[1] == '[':
self.type = self.type + token[1] self.type = self.type + token[1]
token = self.token() token = self.token()
@ -1844,6 +1958,20 @@ class docBuilder:
pass pass
output.write(" </macro>\n") output.write(" </macro>\n")
def serialize_union(self, output, field, desc):
output.write(" <field name='%s' type='union' info='%s'>\n" % (field[1] , desc))
output.write(" <union>\n")
for f in field[3]:
desc = f[2]
if desc == None:
desc = ''
else:
desc = escape(desc)
output.write(" <field name='%s' type='%s' info='%s'/>\n" % (f[1] , f[0], desc))
output.write(" </union>\n")
output.write(" </field>\n")
def serialize_typedef(self, output, name): def serialize_typedef(self, output, name):
id = self.idx.typedefs[name] id = self.idx.typedefs[name]
if id.info[0:7] == 'struct ': if id.info[0:7] == 'struct ':
@ -1862,7 +1990,10 @@ class docBuilder:
desc = '' desc = ''
else: else:
desc = escape(desc) desc = escape(desc)
output.write(" <field name='%s' type='%s' info='%s'/>\n" % (field[1] , field[0], desc)) if field[0] == "union":
self.serialize_union(output, field, desc)
else:
output.write(" <field name='%s' type='%s' info='%s'/>\n" % (field[1] , field[0], desc))
except: except:
print "Failed to serialize struct %s" % (name) print "Failed to serialize struct %s" % (name)
output.write(" </struct>\n") output.write(" </struct>\n")
@ -1961,6 +2092,8 @@ class docBuilder:
continue continue
if dict.structs.has_key(id): if dict.structs.has_key(id):
continue continue
if dict.unions.has_key(id):
continue
if dict.enums.has_key(id): if dict.enums.has_key(id):
continue continue
output.write(" <exports symbol='%s' type='macro'/>\n" % (id)) output.write(" <exports symbol='%s' type='macro'/>\n" % (id))

View File

@ -174,22 +174,62 @@
</pre> </pre>
<table> <table>
<xsl:for-each select="field"> <xsl:for-each select="field">
<tr> <xsl:choose>
<td> <xsl:when test='@type = "union"'>
<xsl:call-template name="dumptext"> <tr><td>union {</td></tr>
<xsl:with-param name="text" select="@type"/> <tr>
</xsl:call-template> <td><table>
</td> <xsl:for-each select="union/field">
<td><xsl:value-of select="@name"/></td> <tr>
<xsl:if test="@info != ''"> <td>
<td> <xsl:call-template name="dumptext">
<xsl:text> : </xsl:text> <xsl:with-param name="text" select="@type"/>
<xsl:call-template name="dumptext"> </xsl:call-template>
<xsl:with-param name="text" select="@info"/> </td>
</xsl:call-template> <td><xsl:value-of select="@name"/></td>
</td> <xsl:if test="@info != ''">
</xsl:if> <td>
</tr> <xsl:text> : </xsl:text>
<xsl:call-template name="dumptext">
<xsl:with-param name="text" select="@info"/>
</xsl:call-template>
</td>
</xsl:if>
</tr>
</xsl:for-each>
</table></td>
<td></td></tr>
<tr><td>}</td>
<td><xsl:value-of select="@name"/></td>
<xsl:if test="@info != ''">
<td>
<xsl:text> : </xsl:text>
<xsl:call-template name="dumptext">
<xsl:with-param name="text" select="@info"/>
</xsl:call-template>
</td>
</xsl:if>
<td></td></tr>
</xsl:when>
<xsl:otherwise>
<tr>
<td>
<xsl:call-template name="dumptext">
<xsl:with-param name="text" select="@type"/>
</xsl:call-template>
</td>
<td><xsl:value-of select="@name"/></td>
<xsl:if test="@info != ''">
<td>
<xsl:text> : </xsl:text>
<xsl:call-template name="dumptext">
<xsl:with-param name="text" select="@info"/>
</xsl:call-template>
</td>
</xsl:if>
</tr>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each> </xsl:for-each>
<xsl:if test="not(field)"> <xsl:if test="not(field)">
<tr> <tr>