mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-15 20:05:40 +00:00
2274 lines
68 KiB
Text
2274 lines
68 KiB
Text
|
#!/usr/bin/perl -w
|
||
|
#
|
||
|
# gtk-doc - GTK DocBook documentation generator.
|
||
|
# Copyright (C) 1998 Damon Chaplin
|
||
|
#
|
||
|
# This program is free software; you can redistribute it and/or modify
|
||
|
# it under the terms of the GNU General Public License as published by
|
||
|
# the Free Software Foundation; either version 2 of the License, or
|
||
|
# (at your option) any later version.
|
||
|
#
|
||
|
# This program is distributed in the hope that it will be useful,
|
||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
# GNU General Public License for more details.
|
||
|
#
|
||
|
# You should have received a copy of the GNU General Public License
|
||
|
# along with this program; if not, write to the Free Software
|
||
|
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||
|
#
|
||
|
|
||
|
#############################################################################
|
||
|
# Script : gtkdoc-mkdb
|
||
|
# Description : This creates the DocBook files from the edited templates.
|
||
|
#
|
||
|
# NOTE: When creating SGML IDS, we append -CAPS to all
|
||
|
# all-caps identifiers to prevent name clashes. (It basically
|
||
|
# never is the case that mixed-case identifiers would collide.)
|
||
|
# See the CreateValidSGMLID function.
|
||
|
#############################################################################
|
||
|
|
||
|
use strict;
|
||
|
use Getopt::Long;
|
||
|
|
||
|
# Options
|
||
|
|
||
|
# name of documentation module
|
||
|
my $MODULE;
|
||
|
my $TMPL_DIR;
|
||
|
my $SGML_OUTPUT_DIR;
|
||
|
my @SOURCE_DIRS;
|
||
|
|
||
|
my %optctl = (module => \$MODULE,
|
||
|
'source-dir' => \@SOURCE_DIRS,
|
||
|
'output-dir' => \$SGML_OUTPUT_DIR,
|
||
|
'tmpl-dir' => \$TMPL_DIR);
|
||
|
GetOptions(\%optctl, "module=s", "source-dir:s", "output-dir:s");
|
||
|
|
||
|
my $ROOT_DIR = ".";
|
||
|
|
||
|
# All the files are written in subdirectories beneath here.
|
||
|
$TMPL_DIR = $TMPL_DIR ? $TMPL_DIR : "$ROOT_DIR/tmpl";
|
||
|
|
||
|
# This is where we put all the DocBook output.
|
||
|
$SGML_OUTPUT_DIR = $SGML_OUTPUT_DIR ? $SGML_OUTPUT_DIR : "$ROOT_DIR/sgml";
|
||
|
|
||
|
# This file contains the object hierarchy.
|
||
|
my $OBJECT_TREE_FILE = "$ROOT_DIR/$MODULE.hierarchy";
|
||
|
|
||
|
# This file contains signal arguments and names.
|
||
|
my $SIGNALS_FILE = "$ROOT_DIR/$MODULE.signals";
|
||
|
|
||
|
# The file containing Arg information.
|
||
|
my $ARGS_FILE = "$ROOT_DIR/$MODULE.args";
|
||
|
|
||
|
# These global arrays store information on signals. Each signal has an entry
|
||
|
# in each of these arrays at the same index, like a multi-dimensional array.
|
||
|
my @SignalObjects; # The GtkObject which emits the signal.
|
||
|
my @SignalNames; # The signal name.
|
||
|
my @SignalReturns; # The return type.
|
||
|
my @SignalPrototypes; # The rest of the prototype of the signal handler.
|
||
|
|
||
|
# These global arrays store information on Args. Each Arg has an entry
|
||
|
# in each of these arrays at the same index, like a multi-dimensional array.
|
||
|
my @ArgObjects; # The GtkObject which has the Arg.
|
||
|
my @ArgNames; # The Arg name.
|
||
|
my @ArgTypes; # The Arg type - gint, GtkArrowType etc.
|
||
|
my @ArgFlags; # How the Arg can be used - readable/writable etc.
|
||
|
|
||
|
# These global hashes store declaration info keyed on a symbol name.
|
||
|
my %Declarations;
|
||
|
my %DeclarationTypes;
|
||
|
my %DeclarationConditional;
|
||
|
my %DeclarationOutput;
|
||
|
|
||
|
# These global hashes store the existing documentation.
|
||
|
my %SymbolDocs;
|
||
|
my %SymbolTypes;
|
||
|
my %SymbolParams;
|
||
|
|
||
|
# These global hashes store documentation scanned from the source files.
|
||
|
my %SourceSymbolDocs;
|
||
|
my %SourceSymbolParams;
|
||
|
|
||
|
# These global arrays store GtkObject and subclasses and the hierarchy.
|
||
|
my @Objects;
|
||
|
my @ObjectLevels;
|
||
|
|
||
|
|
||
|
# Create the root DocBook output directory if it doens't exist.
|
||
|
if (! -e $SGML_OUTPUT_DIR) {
|
||
|
mkdir ("$SGML_OUTPUT_DIR", 0777)
|
||
|
|| die "Can't create directory: $SGML_OUTPUT_DIR";
|
||
|
}
|
||
|
|
||
|
# Function and other declaration output settings.
|
||
|
my $RETURN_TYPE_FIELD_WIDTH = 12;
|
||
|
my $SYMBOL_FIELD_WIDTH = 32;
|
||
|
my $SIGNAL_FIELD_WIDTH = 12;
|
||
|
|
||
|
&ReadSignalsFile ($SIGNALS_FILE);
|
||
|
&ReadArgsFile ($ARGS_FILE);
|
||
|
&ReadObjectHierarchy;
|
||
|
|
||
|
# FIXME: this is the header file output at the top of the Synopsis.
|
||
|
# We should allow this to be changed in the MODULE-sections.txt file.
|
||
|
# gnome.h includes gtk/gtk.h which includes gdk/gdk.h which includes glib.h
|
||
|
# so what should we output? - alternatives?
|
||
|
my $HEADER_FILE = "";
|
||
|
if ($MODULE eq 'glib') {
|
||
|
$HEADER_FILE = "glib.h";
|
||
|
} elsif ($MODULE eq 'gdk') {
|
||
|
$HEADER_FILE = "gtk/gdk.h";
|
||
|
} elsif ($MODULE eq 'gtk') {
|
||
|
$HEADER_FILE = "gtk/gtk.h";
|
||
|
} elsif ($MODULE eq 'gnome' || $MODULE eq 'gnomeui') {
|
||
|
$HEADER_FILE = "gnome.h";
|
||
|
}
|
||
|
|
||
|
for my $dir (@SOURCE_DIRS) {
|
||
|
&ReadSourceDocumentation ($dir);
|
||
|
}
|
||
|
|
||
|
&OutputSGML ("$ROOT_DIR/$MODULE-sections.txt");
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputObjectList
|
||
|
# Description : This outputs the alphabetical list of objects, in a columned
|
||
|
# table. FIXME: Currently this also outputs ancestor objects
|
||
|
# which may not actually be in this module.
|
||
|
# Arguments : none
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputObjectList {
|
||
|
my $cols = 3;
|
||
|
|
||
|
open (OUTPUT, ">$SGML_OUTPUT_DIR/object_index.sgml")
|
||
|
|| die "Can't create $SGML_OUTPUT_DIR/object_index.sgml";
|
||
|
print (OUTPUT <<EOF);
|
||
|
<informaltable pgwide=1 frame="none">
|
||
|
<tgroup cols="$cols">
|
||
|
<colspec colwidth="1*">
|
||
|
<colspec colwidth="1*">
|
||
|
<colspec colwidth="1*">
|
||
|
<tbody>
|
||
|
EOF
|
||
|
|
||
|
my $count = 0;
|
||
|
my $object;
|
||
|
foreach $object (sort(@Objects)) {
|
||
|
my $xref = &MakeXRef ($object);
|
||
|
if ($count % $cols == 0) { print (OUTPUT "<row>\n"); }
|
||
|
print (OUTPUT "<entry>$xref</entry>\n");
|
||
|
if ($count % $cols == ($cols - 1)) { print (OUTPUT "</row>\n"); }
|
||
|
$count++;
|
||
|
}
|
||
|
|
||
|
print (OUTPUT <<EOF);
|
||
|
</tbody></tgroup></informaltable>
|
||
|
EOF
|
||
|
close (OUTPUT);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputSGML
|
||
|
# Description : This collects the output for each section of the docs, and
|
||
|
# outputs each file when the end of the section is found.
|
||
|
# Arguments : $file - the $MODULE-sections.txt file which contains all of
|
||
|
# the functions/macros/structs etc. being documented, organised
|
||
|
# into sections and subsections.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputSGML {
|
||
|
my ($file) = @_;
|
||
|
|
||
|
open (INPUT, $file)
|
||
|
|| die "Can't open $file";
|
||
|
my $book_top = "";
|
||
|
my $book_bottom = "";
|
||
|
my $includes = "";
|
||
|
my $section_includes = "";
|
||
|
my $in_section = 0;
|
||
|
my $title = "";
|
||
|
my $subsection = "";
|
||
|
my $synopsis;
|
||
|
my $details;
|
||
|
my $num_symbols;
|
||
|
while (<INPUT>) {
|
||
|
if (m/^#/) {
|
||
|
next;
|
||
|
|
||
|
} elsif (m/^<SECTION>/) {
|
||
|
$synopsis = "";
|
||
|
$details = "";
|
||
|
$num_symbols = 0;
|
||
|
$in_section = 1;
|
||
|
|
||
|
} elsif (m/^<SUBSECTION\s*(.*)>/i) {
|
||
|
$synopsis .= "\n";
|
||
|
$subsection = $1;
|
||
|
|
||
|
} elsif (m/^<SUBSECTION>/) {
|
||
|
|
||
|
} elsif (m/^<TITLE>(.*)<\/TITLE>/) {
|
||
|
$title = $1;
|
||
|
# print "Section: $title\n";
|
||
|
|
||
|
# We don't want warnings if object & class structs aren't used.
|
||
|
$DeclarationOutput{$title} = 1;
|
||
|
$DeclarationOutput{"${title}Class"} = 1;
|
||
|
|
||
|
} elsif (m/^<FILE>(.*)<\/FILE>/) {
|
||
|
$file = $1;
|
||
|
%SymbolDocs = ();
|
||
|
%SymbolTypes = ();
|
||
|
%SymbolParams = ();
|
||
|
&ReadTemplateFile ("$TMPL_DIR/$file.sgml", 1);
|
||
|
&MergeSourceDocumentation;
|
||
|
|
||
|
} elsif (m/^<INCLUDE>(.*)<\/INCLUDE>/) {
|
||
|
if ($in_section) {
|
||
|
$section_includes = $1;
|
||
|
} else {
|
||
|
$includes = $1;
|
||
|
}
|
||
|
|
||
|
} elsif (m/^<\/SECTION>/) {
|
||
|
if ($title eq "") {
|
||
|
$title = $file;
|
||
|
}
|
||
|
# print "End of section: $title\n";
|
||
|
|
||
|
$file =~ s/\s/_/g;
|
||
|
$file .= ".sgml";
|
||
|
|
||
|
# GtkObjects use their class name as the ID.
|
||
|
my $section_id;
|
||
|
if (&CheckIsObject ($title)) {
|
||
|
$section_id = &CreateValidSGMLID ($title);
|
||
|
} else {
|
||
|
$section_id = &CreateValidSGMLID ("$MODULE-$title");
|
||
|
}
|
||
|
|
||
|
if ($num_symbols > 0) {
|
||
|
$book_top .= "<!entity $section_id SYSTEM \"sgml/$file\">\n";
|
||
|
$book_bottom .= " &$section_id;\n";
|
||
|
|
||
|
if ($section_includes eq "") {
|
||
|
$section_includes = $includes;
|
||
|
}
|
||
|
|
||
|
&OutputSGMLFile ($file, $title, $section_id, $section_includes,
|
||
|
\$synopsis, \$details);
|
||
|
}
|
||
|
$title = "";
|
||
|
$subsection = "";
|
||
|
$in_section = 0;
|
||
|
$section_includes = "";
|
||
|
|
||
|
} elsif (m/^(\S+)/) {
|
||
|
my $symbol = $1;
|
||
|
#print " Symbol: $symbol\n";
|
||
|
|
||
|
my $declaration = $Declarations{$1};
|
||
|
if (defined ($declaration)) {
|
||
|
# We don't want standard macros/functions of GtkObjects,
|
||
|
# or private declarations.
|
||
|
if ($subsection ne "Standard" && $subsection ne "Private") {
|
||
|
my ($synop, $desc) = &OutputDeclaration ($symbol,
|
||
|
$declaration);
|
||
|
$synopsis .= $synop;
|
||
|
$details .= $desc;
|
||
|
}
|
||
|
|
||
|
# Note that the declaration has been output.
|
||
|
$DeclarationOutput{$symbol} = 1;
|
||
|
} else {
|
||
|
print "WARNING: No declaration for: $1\n";
|
||
|
}
|
||
|
$num_symbols++;
|
||
|
}
|
||
|
}
|
||
|
close (INPUT);
|
||
|
|
||
|
&OutputBook ($book_top, $book_bottom);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputDeclaration
|
||
|
# Description : Returns the synopsis and detailed description DocBook
|
||
|
# describing one function/macro etc.
|
||
|
# Arguments : $symbol - the name of the function/macro begin described.
|
||
|
# $declaration - the declaration of the function/macro.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputDeclaration {
|
||
|
my ($symbol, $declaration) = @_;
|
||
|
|
||
|
my $type = $DeclarationTypes {$symbol};
|
||
|
if ($type eq 'MACRO') {
|
||
|
return &OutputMacro ($symbol, $declaration);
|
||
|
} elsif ($type eq 'TYPEDEF') {
|
||
|
return &OutputTypedef ($symbol, $declaration);
|
||
|
} elsif ($type eq 'STRUCT') {
|
||
|
return &OutputStruct ($symbol, $declaration);
|
||
|
} elsif ($type eq 'ENUM') {
|
||
|
return &OutputEnum ($symbol, $declaration);
|
||
|
} elsif ($type eq 'UNION') {
|
||
|
return &OutputUnion ($symbol, $declaration);
|
||
|
} elsif ($type eq 'VARIABLE') {
|
||
|
return &OutputVariable ($symbol, $declaration);
|
||
|
|
||
|
} elsif ($type eq 'FUNCTION') {
|
||
|
return &OutputFunction ($symbol, $declaration, $type);
|
||
|
} elsif ($type eq 'USER_FUNCTION') {
|
||
|
return &OutputFunction ($symbol, $declaration, $type);
|
||
|
} else {
|
||
|
die "Unknown symbol type";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputMacro
|
||
|
# Description : Returns the synopsis and detailed description of a macro.
|
||
|
# Arguments : $symbol - the macro.
|
||
|
# $declaration - the declaration of the macro.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputMacro {
|
||
|
my ($symbol, $declaration) = @_;
|
||
|
my $id = &CreateValidSGMLID ($symbol);
|
||
|
my $synop = "#define <link linkend=\"$id\">$symbol</link>";
|
||
|
my $desc;
|
||
|
my $args = "";
|
||
|
if ($declaration =~ m/^\s*#\s*define\s+\w+(\([^\)]*\))/) {
|
||
|
$args = $1;
|
||
|
|
||
|
if (length ($symbol) < $SYMBOL_FIELD_WIDTH) {
|
||
|
$synop .= (' ' x ($SYMBOL_FIELD_WIDTH - length ($symbol)));
|
||
|
}
|
||
|
|
||
|
$synop .= &CreateValidSGML ($args);
|
||
|
}
|
||
|
$synop .= "\n";
|
||
|
|
||
|
if ($args ne "") {
|
||
|
$desc = "<refsect2>\n<title><anchor id=\"$id\">${symbol}()</title>\n";
|
||
|
} else {
|
||
|
$desc = "<refsect2>\n<title><anchor id=\"$id\">$symbol</title>\n";
|
||
|
}
|
||
|
# Don't output the macro definition if is is a conditional macro or it
|
||
|
# looks like a function, i.e. starts with "g_" or "_?gnome_", or it is
|
||
|
# longer than 2 lines, otherwise we get lots of complicated macros like
|
||
|
# g_assert.
|
||
|
if (!defined ($DeclarationConditional{$symbol}) && ($symbol !~ m/^g_/)
|
||
|
&& ($symbol !~ m/^_?gnome_/) && (($declaration =~ tr/\n//) < 2)) {
|
||
|
$declaration = &CreateValidSGML ($declaration);
|
||
|
$desc .= "<programlisting>$declaration</programlisting>\n";
|
||
|
} else {
|
||
|
$desc .= "<programlisting>#define $symbol";
|
||
|
$desc .= &CreateValidSGML ($args);
|
||
|
$desc .= "</programlisting>\n";
|
||
|
}
|
||
|
if (defined ($SymbolDocs{$symbol})) {
|
||
|
$desc .= &ExpandAbbreviations($SymbolDocs{$symbol});
|
||
|
}
|
||
|
$desc .= &OutputParamDescriptions ("MACRO", $symbol);
|
||
|
$desc .= "</refsect2>\n";
|
||
|
return ($synop, $desc);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputTypedef
|
||
|
# Description : Returns the synopsis and detailed description of a typedef.
|
||
|
# Arguments : $symbol - the typedef.
|
||
|
# $declaration - the declaration of the typedef,
|
||
|
# e.g. 'typedef unsigned int guint;'
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputTypedef {
|
||
|
my ($symbol, $declaration) = @_;
|
||
|
my $id = &CreateValidSGMLID ($symbol);
|
||
|
my $synop = "typedef <link linkend=\"$id\">$symbol</link>;\n";
|
||
|
my $desc = "<refsect2>\n<title><anchor id=\"$id\">$symbol</title>\n";
|
||
|
if (!defined ($DeclarationConditional{$symbol})) {
|
||
|
$declaration = &CreateValidSGML ($declaration);
|
||
|
$desc .= "<programlisting>$declaration</programlisting>\n";
|
||
|
}
|
||
|
if (defined ($SymbolDocs{$symbol})) {
|
||
|
$desc .= &ExpandAbbreviations($SymbolDocs{$symbol});
|
||
|
}
|
||
|
$desc .= "</refsect2>\n";
|
||
|
return ($synop, $desc);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputStruct
|
||
|
# Description : Returns the synopsis and detailed description of a struct.
|
||
|
# We check if it is a widget struct, and if so we only output
|
||
|
# parts of it that are noted as public fields.
|
||
|
# We also use a different SGML ID for widget structs, since the
|
||
|
# original ID is used for the entire RefEntry.
|
||
|
# Arguments : $symbol - the struct.
|
||
|
# $declaration - the declaration of the struct.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputStruct {
|
||
|
my ($symbol, $declaration) = @_;
|
||
|
|
||
|
my $is_widget_struct = 0;
|
||
|
if (&CheckIsObject ($symbol)) {
|
||
|
# print "Found widget struct: $symbol\n";
|
||
|
$is_widget_struct = 1;
|
||
|
}
|
||
|
|
||
|
my $id;
|
||
|
if ($is_widget_struct) {
|
||
|
$id = &CreateValidSGMLID ($symbol . "_struct");
|
||
|
} else {
|
||
|
$id = &CreateValidSGMLID ($symbol);
|
||
|
}
|
||
|
my $synop = "struct <link linkend=\"$id\">$symbol</link>;\n";
|
||
|
my $desc = "<refsect2>\n<title><anchor id=\"$id\">struct $symbol</title>\n";
|
||
|
|
||
|
# Form a pretty-printed, private-data-removed form of the declaration
|
||
|
|
||
|
my $decl_out;
|
||
|
if ($declaration =~ m/^\s*$/) {
|
||
|
# print "Found opaque struct\n";
|
||
|
$decl_out = "struct $symbol;";
|
||
|
} elsif ($is_widget_struct) {
|
||
|
my $public = 0;
|
||
|
my $new_declaration = "";
|
||
|
my $decl_line;
|
||
|
foreach $decl_line (split (/\n/, $declaration)) {
|
||
|
# print "Struct line: $decl_line\n";
|
||
|
if ($decl_line =~ m%/\*\s*<\s*public\s*>\s*\*/%) {
|
||
|
$public = 1;
|
||
|
} elsif ($decl_line =~ m%/\*\s*<\s*private\s*>\s*\*/%) {
|
||
|
$public = 0;
|
||
|
} elsif ($public) {
|
||
|
$new_declaration .= $decl_line . "\n";
|
||
|
}
|
||
|
}
|
||
|
if ($new_declaration) {
|
||
|
$decl_out = "struct $symbol {\n" . $new_declaration;
|
||
|
# If we finished with public set, we already have the struct end.
|
||
|
if ($public == 0) {
|
||
|
$decl_out .= "};\n";
|
||
|
}
|
||
|
} else {
|
||
|
$decl_out = "struct $symbol;";
|
||
|
}
|
||
|
} else {
|
||
|
$decl_out = $declaration;
|
||
|
}
|
||
|
|
||
|
$decl_out = &CreateValidSGML ($decl_out);
|
||
|
$desc .= "<programlisting>$decl_out</programlisting>\n";
|
||
|
if (defined ($SymbolDocs{$symbol})) {
|
||
|
$desc .= &ExpandAbbreviations($SymbolDocs{$symbol});
|
||
|
}
|
||
|
|
||
|
# Create a table of fields and descriptions
|
||
|
|
||
|
# FIXME: Inserting  's into the produced type declarations here would
|
||
|
# improve the output in most situations ... except for function
|
||
|
# members of structs!
|
||
|
my @fields = ParseStructDeclaration($declaration, $is_widget_struct, \&MakeXRef,
|
||
|
sub {
|
||
|
"<structfield>$_[0]</structfield>";
|
||
|
});
|
||
|
my $params = $SymbolParams{$symbol};
|
||
|
|
||
|
# If no parameters are filled in, we don't generate the description
|
||
|
# table, for backwards compatibility
|
||
|
|
||
|
my $found = 0;
|
||
|
if (defined $params) {
|
||
|
for (my $i = 1; $i <= $#$params; $i += 2) {
|
||
|
if ($params->[$i] =~ /\S/) {
|
||
|
$found = 1;
|
||
|
last;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($found) {
|
||
|
my %field_descrs = @$params;
|
||
|
|
||
|
$desc .= <<EOF;
|
||
|
<informaltable pgwide=1 frame="none" role="struct">
|
||
|
<tgroup cols="2">
|
||
|
<colspec colwidth="2*">
|
||
|
<colspec colwidth="8*">
|
||
|
<tbody>
|
||
|
EOF
|
||
|
while (@fields) {
|
||
|
my $field_name = shift @fields;
|
||
|
my $text = shift @fields;
|
||
|
my $field_descr = $field_descrs{$field_name};
|
||
|
|
||
|
$desc .= "<row>\n<entry>$text</entry>\n";
|
||
|
if (defined $field_descr) {
|
||
|
$desc .= "<entry>".&ExpandAbbreviations($field_descr)."</entry>\n";
|
||
|
} else {
|
||
|
$desc .= "<entry></entry>\n";
|
||
|
}
|
||
|
$desc .= "</row>\n";
|
||
|
}
|
||
|
|
||
|
$desc .= "</tbody></tgroup></informaltable>";
|
||
|
}
|
||
|
$desc .= "</refsect2>\n";
|
||
|
return ($synop, $desc);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputEnum
|
||
|
# Description : Returns the synopsis and detailed description of a enum.
|
||
|
# Arguments : $symbol - the enum.
|
||
|
# $declaration - the declaration of the enum.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputEnum {
|
||
|
my ($symbol, $declaration) = @_;
|
||
|
my $id = &CreateValidSGMLID ($symbol);
|
||
|
my $synop = "enum <link linkend=\"$id\">$symbol</link>;\n";
|
||
|
my $desc = "<refsect2>\n<title><anchor id=\"$id\">enum $symbol</title>\n";
|
||
|
$declaration = &CreateValidSGML ($declaration);
|
||
|
$desc .= "<programlisting>$declaration</programlisting>\n";
|
||
|
if (defined ($SymbolDocs{$symbol})) {
|
||
|
$desc .= &ExpandAbbreviations($SymbolDocs{$symbol});
|
||
|
}
|
||
|
|
||
|
# Create a table of fields and descriptions
|
||
|
|
||
|
my @members = ParseEnumDeclaration($declaration);
|
||
|
my $params = $SymbolParams{$symbol};
|
||
|
|
||
|
# If no parameters are filled in, we don't generate the description
|
||
|
# table, for backwards compatibility
|
||
|
|
||
|
my $found = 0;
|
||
|
if (defined $params) {
|
||
|
for (my $i = 1; $i <= $#$params; $i += 2) {
|
||
|
if ($params->[$i] =~ /\S/) {
|
||
|
$found = 1;
|
||
|
last;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($found) {
|
||
|
my %member_descrs = @$params;
|
||
|
|
||
|
$desc .= <<EOF;
|
||
|
<informaltable pgwide=1 frame="none" role="enum">
|
||
|
<tgroup cols="2">
|
||
|
<colspec colwidth="2*">
|
||
|
<colspec colwidth="8*">
|
||
|
<tbody>
|
||
|
EOF
|
||
|
for my $member_name (@members) {
|
||
|
my $member_descr = $member_descrs{$member_name};
|
||
|
|
||
|
$desc .= "<row>\n<entry><literal>$member_name</literal></entry>\n";
|
||
|
if (defined $member_descr) {
|
||
|
$desc .= "<entry>".&ExpandAbbreviations($member_descr)."</entry>\n";
|
||
|
} else {
|
||
|
$desc .= "<entry></entry>\n";
|
||
|
}
|
||
|
$desc .= "</row>\n";
|
||
|
}
|
||
|
|
||
|
$desc .= "</tbody></tgroup></informaltable>";
|
||
|
}
|
||
|
|
||
|
$desc .= "</refsect2>\n";
|
||
|
return ($synop, $desc);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputUnion
|
||
|
# Description : Returns the synopsis and detailed description of a union.
|
||
|
# Arguments : $symbol - the union.
|
||
|
# $declaration - the declaration of the union.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputUnion {
|
||
|
my ($symbol, $declaration) = @_;
|
||
|
my $id = &CreateValidSGMLID ($symbol);
|
||
|
my $synop = "union <link linkend=\"$id\">$symbol</link>;\n";
|
||
|
my $desc = "<refsect2>\n<title><anchor id=\"$id\">union $symbol</title>\n";
|
||
|
$declaration = &CreateValidSGML ($declaration);
|
||
|
$desc .= "<programlisting>$declaration</programlisting>\n";
|
||
|
if (defined ($SymbolDocs{$symbol})) {
|
||
|
$desc .= &ExpandAbbreviations($SymbolDocs{$symbol});
|
||
|
}
|
||
|
$desc .= "</refsect2>\n";
|
||
|
return ($synop, $desc);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputVariable
|
||
|
# Description : Returns the synopsis and detailed description of a variable.
|
||
|
# Arguments : $symbol - the extern'ed variable.
|
||
|
# $declaration - the declaration of the variable.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputVariable {
|
||
|
my ($symbol, $declaration) = @_;
|
||
|
my $id = &CreateValidSGMLID ($symbol);
|
||
|
|
||
|
my $synop;
|
||
|
if ($declaration =~ m/^\s*extern\s+((const\s+|unsigned\s+)*\w+)(\s+\*+|\*+|\s)(\s*)([A-Za-z]\w*)\s*;/) {
|
||
|
my $mod = defined ($1) ? $1 : "";
|
||
|
my $ptr = defined ($3) ? $3 : "";
|
||
|
my $space = defined ($4) ? $4 : "";
|
||
|
$synop = "extern $mod$ptr$space<link linkend=\"$id\">$symbol</link>;\n";
|
||
|
|
||
|
} else {
|
||
|
$synop = "extern <link linkend=\"$id\">$symbol</link>;\n";
|
||
|
}
|
||
|
|
||
|
my $desc = "<refsect2>\n<title><anchor id=\"$id\">$symbol</title>\n";
|
||
|
$declaration = &CreateValidSGML ($declaration);
|
||
|
$desc .= "<programlisting>$declaration</programlisting>\n";
|
||
|
if (defined ($SymbolDocs{$symbol})) {
|
||
|
$desc .= &ExpandAbbreviations($SymbolDocs{$symbol});
|
||
|
}
|
||
|
$desc .= "</refsect2>\n";
|
||
|
return ($synop, $desc);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputFunction
|
||
|
# Description : Returns the synopsis and detailed description of a function.
|
||
|
# Arguments : $symbol - the function.
|
||
|
# $declaration - the declaration of the function.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputFunction {
|
||
|
my ($symbol, $declaration, $symbol_type) = @_;
|
||
|
my $id = &CreateValidSGMLID ($symbol);
|
||
|
|
||
|
# Take out the return type
|
||
|
$declaration =~ s/<RETURNS>\s*(const\s+|unsigned\s+)*(\w+)\s*(\**)\s*<\/RETURNS>\n//;
|
||
|
my $type_modifier = defined($1) ? $1 : "";
|
||
|
my $type = $2;
|
||
|
my $pointer = $3;
|
||
|
my $xref = &MakeXRef ($type);
|
||
|
my $start = "";
|
||
|
if ($symbol_type eq 'USER_FUNCTION') {
|
||
|
# $start = "typedef ";
|
||
|
}
|
||
|
|
||
|
my $ret_type_len = length ($start) + length ($type_modifier)
|
||
|
+ length ($pointer) + length ($type);
|
||
|
my $ret_type_output;
|
||
|
my $symbol_len;
|
||
|
if ($ret_type_len < $RETURN_TYPE_FIELD_WIDTH) {
|
||
|
$ret_type_output = "$start$type_modifier$xref$pointer"
|
||
|
. (' ' x ($RETURN_TYPE_FIELD_WIDTH - $ret_type_len));
|
||
|
$symbol_len = 0;
|
||
|
} else {
|
||
|
# $ret_type_output = "$start$type_modifier$xref$pointer\n"
|
||
|
# . (' ' x $RETURN_TYPE_FIELD_WIDTH);
|
||
|
|
||
|
$ret_type_output = "$start$type_modifier$xref$pointer ";
|
||
|
$symbol_len = $ret_type_len + 1 - $RETURN_TYPE_FIELD_WIDTH;
|
||
|
}
|
||
|
|
||
|
$symbol_len += length ($symbol);
|
||
|
my $char1 = my $char2 = my $char3 = "";
|
||
|
if ($symbol_type eq 'USER_FUNCTION') {
|
||
|
$symbol_len += 3;
|
||
|
$char1 = "(";
|
||
|
$char2 = "*";
|
||
|
$char3 = ")";
|
||
|
}
|
||
|
|
||
|
my ($symbol_output, $symbol_desc_output);
|
||
|
if ($symbol_len < $SYMBOL_FIELD_WIDTH) {
|
||
|
$symbol_output = "$char1<link linkend=\"$id\">$char2$symbol</link>$char3"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH - $symbol_len));
|
||
|
$symbol_desc_output = "$char1$char2$symbol$char3"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH - $symbol_len));
|
||
|
} else {
|
||
|
$symbol_output = "$char1<link linkend=\"$id\">$char2$symbol</link>$char3\n"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH));
|
||
|
$symbol_desc_output = "$char1$char2$symbol$char3\n"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH));
|
||
|
}
|
||
|
|
||
|
my $synop = $ret_type_output . $symbol_output . '(';
|
||
|
my $desc = "<refsect2>\n<title><anchor id=\"$id\">${symbol} ()</title>\n";
|
||
|
$desc .= "<programlisting>${ret_type_output}$symbol_desc_output(";
|
||
|
|
||
|
my $param_num = 0;
|
||
|
while ($declaration ne "") {
|
||
|
if ($declaration =~ s/^[\s,]+//) {
|
||
|
# skip whitespace and commas
|
||
|
next;
|
||
|
|
||
|
} elsif ($declaration =~ s/^void\s*[,\n]//) {
|
||
|
$synop .= "void";
|
||
|
$desc .= "void";
|
||
|
|
||
|
} elsif ($declaration =~ s/^...\s*[,\n]//) {
|
||
|
if ($param_num == 0) {
|
||
|
$synop .= "...";
|
||
|
$desc .= "...";
|
||
|
} else {
|
||
|
$synop .= ",\n"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH))
|
||
|
. " ...";
|
||
|
$desc .= ",\n"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH))
|
||
|
. " ...";
|
||
|
}
|
||
|
|
||
|
# allow alphanumerics, '_', '[' & ']' in param names
|
||
|
} elsif ($declaration =~ s/^(const\s+|unsigned\s+)*(struct\s+)?(\w+)\s*(\**)\s*(const\s+)?(\**)?\s*(\w+)?\s*(\[\d*\])?\s*[,\n]//) {
|
||
|
my $mod1 = defined($1) ? $1 : "";
|
||
|
if (defined($2)) { $mod1 .= $2; }
|
||
|
my $type = $3;
|
||
|
my $ptr1 = $4;
|
||
|
my $mod2 = defined($5) ? $5 : "";
|
||
|
my $ptr2 = $6;
|
||
|
my $name = defined($7) ? $7 : "";
|
||
|
if ($name) { $ptr1 = " " . $ptr1; }
|
||
|
my $array = defined($8) ? $8 : "";
|
||
|
my $xref = &MakeXRef ($type);
|
||
|
|
||
|
# print "Type: $mod1$type $ptr1 $mod2 $name $array\n";
|
||
|
if ($param_num == 0) {
|
||
|
$synop .= "$mod1$xref$ptr1$mod2$ptr2$name$array";
|
||
|
$desc .= "$mod1$xref$ptr1$mod2$ptr2$name$array";
|
||
|
} else {
|
||
|
$synop .= ",\n"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH))
|
||
|
. " $mod1$xref$ptr1$mod2$ptr2$name$array";
|
||
|
$desc .= ",\n"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH))
|
||
|
. " $mod1$xref$ptr1$mod2$ptr2$name$array";
|
||
|
}
|
||
|
|
||
|
# Try to match parameters which are functions.
|
||
|
} elsif ($declaration =~ s/^(const\s+|unsigned\s+)*(struct\s+)?(\w+)\s*(\**)\s*(const\s+)?\(\s*\*\s*(\w+)\s*\)\s*\(([^)]*)\)\s*[,\n]//) {
|
||
|
my $mod1 = defined($1) ? $1 : "";
|
||
|
if (defined($2)) { $mod1 .= $2; }
|
||
|
my $type = $3;
|
||
|
my $ptr1 = $4;
|
||
|
my $mod2 = defined($5) ? $5 : "";
|
||
|
my $name = $6;
|
||
|
my $func_params = $7;
|
||
|
my $xref = &MakeXRef ($type);
|
||
|
|
||
|
# print "Type: $mod1$type$ptr1$mod2(*$name)($func_params)\n";
|
||
|
if ($param_num == 0) {
|
||
|
$synop .= "$mod1$xref$ptr1$mod2 (*$name) ($func_params)";
|
||
|
$desc .= "$mod1$xref$ptr1$mod2 (*$name) ($func_params)";
|
||
|
} else {
|
||
|
$synop .= ",\n"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH))
|
||
|
. " $mod1$xref$ptr1$mod2 (*$name) ($func_params)";
|
||
|
$desc .= ",\n"
|
||
|
. (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH))
|
||
|
. " $mod1$xref$ptr1$mod2 (*$name) ($func_params)";
|
||
|
}
|
||
|
|
||
|
} else {
|
||
|
print "###Can't parse args for function $symbol: $declaration\n";
|
||
|
last;
|
||
|
}
|
||
|
$param_num++;
|
||
|
}
|
||
|
$synop .= ");\n";
|
||
|
$desc .= ");</programlisting>\n";
|
||
|
if (defined ($SymbolDocs{$symbol})) {
|
||
|
$desc .= &ExpandAbbreviations($SymbolDocs{$symbol});
|
||
|
}
|
||
|
|
||
|
$desc .= &OutputParamDescriptions ("FUNCTION", $symbol);
|
||
|
$desc .= "</refsect2>\n";
|
||
|
return ($synop, $desc);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputParamDescriptions
|
||
|
# Description : Returns the DocBook output describing the parameters of a
|
||
|
# function, macro or signal handler.
|
||
|
# Arguments : $symbol_type - 'FUNCTION', 'MACRO' or 'SIGNAL'. Signal
|
||
|
# handlers have an implicit user_data parameter last.
|
||
|
# $symbol - the name of the function/macro being described.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputParamDescriptions {
|
||
|
my ($symbol_type, $symbol) = @_;
|
||
|
my $output = "";
|
||
|
|
||
|
if (defined ($SymbolParams{$symbol})) {
|
||
|
my $returns = "";
|
||
|
my $params = $SymbolParams{$symbol};
|
||
|
my $params_desc = "";
|
||
|
if ($#$params < 0) {
|
||
|
print "WARNING: 0 parameters\n";
|
||
|
}
|
||
|
my $j;
|
||
|
for ($j = 0; $j <= $#$params; $j += 2) {
|
||
|
my $param_name = $$params[$j];
|
||
|
my $param = $$params[$j + 1];
|
||
|
if ($param_name eq "Returns") {
|
||
|
$returns = &ExpandAbbreviations($param);
|
||
|
} else {
|
||
|
if ($param_name eq "Varargs") {
|
||
|
$param_name = "...";
|
||
|
}
|
||
|
$param = &ExpandAbbreviations($param);
|
||
|
$params_desc .= "<row><entry align=\"right\"><parameter>$param_name</parameter> :</entry>\n<entry>$param</entry></row>\n";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
# Signals have an implicit user_data parameter which we describe.
|
||
|
if ($symbol_type eq "SIGNAL") {
|
||
|
$params_desc .= "<row><entry align=\"right\"><parameter>user_data</parameter> :</entry>\n<entry>user data set when the signal handler was connected.</entry></row>\n";
|
||
|
}
|
||
|
|
||
|
# Start a table if we need one.
|
||
|
if ($params_desc || $returns) {
|
||
|
$output .= <<EOF;
|
||
|
<informaltable pgwide=1 frame="none" role="params">
|
||
|
<tgroup cols="2">
|
||
|
<colspec colwidth="2*">
|
||
|
<colspec colwidth="8*">
|
||
|
<tbody>
|
||
|
EOF
|
||
|
|
||
|
if ($params_desc ne "") {
|
||
|
# $output .= "<row><entry>Parameters:</entry></row>\n";
|
||
|
$output .= $params_desc;
|
||
|
}
|
||
|
|
||
|
# Output the returns info last.
|
||
|
if ($returns) {
|
||
|
$output .= "<row><entry align=\"right\"><emphasis>Returns</emphasis> :</entry><entry>$returns</entry></row>\n";
|
||
|
}
|
||
|
|
||
|
# Finish the table.
|
||
|
$output .= "</tbody></tgroup></informaltable>";
|
||
|
}
|
||
|
}
|
||
|
return $output;
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputSGMLFile
|
||
|
# Description : Outputs the final DocBook file for one section.
|
||
|
# Arguments : $file - the name of the file.
|
||
|
# $title - the title from the $MODULE-sections.txt file, which
|
||
|
# will be overriden by the title in the template file.
|
||
|
# $section_id - the SGML id to use for the toplevel tag.
|
||
|
# $includes - comma-separates list of include files added at top
|
||
|
# of synopsis, with '<' '>' around them.
|
||
|
# $synopsis - reference to the DocBook for the Synopsis part.
|
||
|
# $details - reference to the DocBook for the Details part.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputSGMLFile {
|
||
|
my ($file, $title, $section_id, $includes, $synopsis, $details) = @_;
|
||
|
|
||
|
# Find out if this is a GtkObject or descendant.
|
||
|
my $signals_synop = "";
|
||
|
my $signals_desc = "";
|
||
|
my $args_synop = "";
|
||
|
my $args_desc = "";
|
||
|
my $hierarchy = "";
|
||
|
if (&CheckIsObject ($title)) {
|
||
|
($signals_synop, $signals_desc) = &GetSignals ($title);
|
||
|
($args_synop, $args_desc) = &GetArgs ($title);
|
||
|
$hierarchy = &GetHierarchy ($title);
|
||
|
}
|
||
|
|
||
|
# The edited title overrides the one from the sections file.
|
||
|
my $new_title = $SymbolDocs{"$TMPL_DIR/$file:Title"};
|
||
|
if (defined ($new_title) && $new_title !~ m/^\s*$/) {
|
||
|
$title = $new_title;
|
||
|
# print "Found title: $title\n";
|
||
|
}
|
||
|
my $short_desc = $SymbolDocs{"$TMPL_DIR/$file:Short_Description"};
|
||
|
if (!defined ($short_desc) || $short_desc =~ m/^\s*$/) {
|
||
|
# $short_desc = "one line description goes here.";
|
||
|
$short_desc = "";
|
||
|
} else {
|
||
|
$short_desc = &ExpandAbbreviations($short_desc);
|
||
|
# print "Found short_desc: $short_desc";
|
||
|
}
|
||
|
my $long_desc = $SymbolDocs{"$TMPL_DIR/$file:Long_Description"};
|
||
|
if (!defined ($long_desc) || $long_desc =~ m/^\s*$/) {
|
||
|
$long_desc = "<para>\nA longer description goes here.\n</para>\n";
|
||
|
} else {
|
||
|
$long_desc = &ExpandAbbreviations($long_desc);
|
||
|
# print "Found long_desc: $long_desc";
|
||
|
}
|
||
|
my $see_also = $SymbolDocs{"$TMPL_DIR/$file:See_Also"};
|
||
|
if (!defined ($see_also) || $see_also =~ m%^\s*(<para>)?\s*(</para>)?\s*$%) {
|
||
|
$see_also = "";
|
||
|
} else {
|
||
|
$see_also = &ExpandAbbreviations($see_also);
|
||
|
# print "Found see_also: $see_also";
|
||
|
}
|
||
|
if ($see_also) {
|
||
|
$see_also = "<refsect1>\n<title>See Also</title>\n$see_also\n</refsect1>\n";
|
||
|
}
|
||
|
|
||
|
my ($sec, $min, $hour, $mday, $mon, $year, $wday, $yday, $isdst) =
|
||
|
gmtime (time);
|
||
|
my $month = (qw(Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec))[$mon];
|
||
|
$year += 1900;
|
||
|
|
||
|
my $include_output = "";
|
||
|
my $include;
|
||
|
foreach $include (split (/,/, $includes)) {
|
||
|
$include_output .= "#include <${include}>\n";
|
||
|
}
|
||
|
|
||
|
open (OUTPUT, ">$SGML_OUTPUT_DIR/$file")
|
||
|
|| die "Can't create $SGML_OUTPUT_DIR/$file";
|
||
|
|
||
|
# Note: The refname and refpurpose are on the same line to stop
|
||
|
# docbook-to-man 1.08 putting them on separate lines.
|
||
|
print OUTPUT <<EOF;
|
||
|
<refentry id="$section_id" revision="$mday $month $year">
|
||
|
<refmeta>
|
||
|
<refentrytitle>$title</refentrytitle>
|
||
|
<manvolnum>3</manvolnum>
|
||
|
<refmiscinfo>\U$MODULE\E Library</refmiscinfo>
|
||
|
</refmeta>
|
||
|
|
||
|
<refnamediv>
|
||
|
<refname>$title</refname><refpurpose>$short_desc</refpurpose>
|
||
|
</refnamediv>
|
||
|
|
||
|
<refsynopsisdiv><title>Synopsis</title>
|
||
|
<synopsis>
|
||
|
|
||
|
$include_output
|
||
|
|
||
|
$${synopsis}</synopsis>
|
||
|
</refsynopsisdiv>
|
||
|
|
||
|
$hierarchy
|
||
|
$args_synop
|
||
|
$signals_synop
|
||
|
|
||
|
<refsect1>
|
||
|
<title>Description</title>
|
||
|
$long_desc
|
||
|
</refsect1>
|
||
|
|
||
|
<refsect1>
|
||
|
<title>Details</title>
|
||
|
$$details
|
||
|
</refsect1>
|
||
|
$args_desc
|
||
|
$signals_desc
|
||
|
|
||
|
$see_also
|
||
|
</refentry>
|
||
|
EOF
|
||
|
close (OUTPUT);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : OutputBook
|
||
|
# Description : Outputs the SGML entities that need to be included into the
|
||
|
# main SGML file for the module.
|
||
|
# Arguments : $book_top - the declarations of the entities, which are added
|
||
|
# at the top of the main SGML file.
|
||
|
# $book_bottom - the references to the entities, which are
|
||
|
# added in the main SGML file at the desired position.
|
||
|
#############################################################################
|
||
|
|
||
|
sub OutputBook {
|
||
|
my ($book_top, $book_bottom) = @_;
|
||
|
|
||
|
open (OUTPUT, ">$SGML_OUTPUT_DIR/$MODULE-doc.top")
|
||
|
|| die "Can't create $SGML_OUTPUT_DIR/$MODULE-doc.top";
|
||
|
print OUTPUT $book_top;
|
||
|
close (OUTPUT);
|
||
|
|
||
|
open (OUTPUT, ">$SGML_OUTPUT_DIR/$MODULE-doc.bottom")
|
||
|
|| die "Can't create $SGML_OUTPUT_DIR/$MODULE-doc.bottom";
|
||
|
print OUTPUT $book_bottom;
|
||
|
close (OUTPUT);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : CreateValidSGMLID
|
||
|
# Description : Creates a valid SGML 'id' from the given string.
|
||
|
# NOTE: SGML ids are case-insensitive, so we have a few special
|
||
|
# cases to avoid clashes of ids.
|
||
|
# Arguments : $id - the string to be converted into a valid SGML id.
|
||
|
#############################################################################
|
||
|
|
||
|
sub CreateValidSGMLID {
|
||
|
my ($id) = $_[0];
|
||
|
|
||
|
# Append -CAPS to all all-caps identifiers
|
||
|
|
||
|
# Special case, '_' would end up as '' so we use 'gettext-macro' instead.
|
||
|
if ($id eq "_") { return "gettext-macro"; }
|
||
|
|
||
|
if ($id !~ /[a-z]/) { $id .= "-CAPS" };
|
||
|
|
||
|
$id =~ s/[_ ]/-/g;
|
||
|
$id =~ s/[,\.]//g;
|
||
|
$id =~ s/^-*//;
|
||
|
$id =~ s/::/-/g;
|
||
|
|
||
|
return $id;
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : CreateValidSGML
|
||
|
# Description : This turns any chars which are used in SGML into entities,
|
||
|
# e.g. '<' into '<'
|
||
|
# Arguments : $text - the text to turn into proper SGML.
|
||
|
#############################################################################
|
||
|
|
||
|
sub CreateValidSGML {
|
||
|
my ($text) = @_;
|
||
|
$text =~ s/&/&/g; # Do this first, or the others get messed up.
|
||
|
$text =~ s/</</g;
|
||
|
$text =~ s/>/>/g;
|
||
|
return $text;
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ExpandAbbreviations
|
||
|
# Description : This turns the abbreviations function(), macro(), @param,
|
||
|
# %constant, and #symbol into appropriate DocBook markup.
|
||
|
# Arguments : $text - the text to expand.
|
||
|
#############################################################################
|
||
|
|
||
|
sub ExpandAbbreviations {
|
||
|
my ($text) = @_;
|
||
|
|
||
|
# Convert 'function()' or 'macro()'
|
||
|
$text =~ s/(\w+)\s*\(\)/&MakeXRef($1) . "()";/eg;
|
||
|
|
||
|
# Convert '@param'
|
||
|
$text =~ s/\@(\w+)/<parameter>$1<\/parameter>/g;
|
||
|
|
||
|
# Convert '%constant'. Also allow negative numbers, e.g. %-1.
|
||
|
$text =~ s/\%(-?\w+)/<literal>$1<\/literal>/g;
|
||
|
|
||
|
# Convert '#symbol'
|
||
|
$text =~ s/#([\w-]+)/&MakeXRef($1);/eg;
|
||
|
|
||
|
return $text;
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : MakeXRef
|
||
|
# Description : This returns a cross-reference link to the given symbol.
|
||
|
# Though it doesn't try to do this for a few standard C types
|
||
|
# that it knows won't be in the documentation.
|
||
|
# Arguments : $symbol - the symbol to try to create a XRef to.
|
||
|
#############################################################################
|
||
|
|
||
|
sub MakeXRef {
|
||
|
my ($symbol) = $_[0];
|
||
|
# print "Getting type link for $symbol\n";
|
||
|
|
||
|
# Don't create a link for some standard C types and functions, to cut
|
||
|
# down on the number of warnings output by jade.
|
||
|
if ($symbol eq "void" || $symbol eq "va_list" || $symbol eq "int"
|
||
|
|| $symbol eq "char" || $symbol eq "printf" || $symbol eq "sprintf") {
|
||
|
return $symbol;
|
||
|
}
|
||
|
|
||
|
my $symbol_id = &CreateValidSGMLID ($symbol);
|
||
|
# Get rid of special '-struct' suffix.
|
||
|
$symbol =~ s/-struct$//;
|
||
|
return "<link linkend=\"$symbol_id\">$symbol</link>";
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : GetHierarchy
|
||
|
# Description : Returns the DocBook output describing the ancestors of a
|
||
|
# GtkObject subclass. It uses the global @Objects and
|
||
|
# @ObjectLevels arrays to walk up the tree.
|
||
|
# Arguments : $object - the GtkObject subclass.
|
||
|
#############################################################################
|
||
|
|
||
|
sub GetHierarchy {
|
||
|
my ($object) = @_;
|
||
|
|
||
|
# Find object in the objects array.
|
||
|
my $found = 0;
|
||
|
my $i;
|
||
|
for ($i = 0; $i < @Objects; $i++) {
|
||
|
if ($Objects[$i] eq $object) {
|
||
|
$found = 1;
|
||
|
last;
|
||
|
}
|
||
|
}
|
||
|
if (!$found) {
|
||
|
return "";
|
||
|
}
|
||
|
|
||
|
# Walk up the hierarchy, pushing ancestors onto the ancestors array.
|
||
|
my @ancestors = ();
|
||
|
push (@ancestors, $object);
|
||
|
my $level = $ObjectLevels[$i];
|
||
|
# print "Level: $level\n";
|
||
|
while ($level > 1) {
|
||
|
$i--;
|
||
|
if ($ObjectLevels[$i] < $level) {
|
||
|
push (@ancestors, $Objects[$i]);
|
||
|
$level = $ObjectLevels[$i];
|
||
|
# print "Level: $level\n";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
# Output the ancestors list, indented and with links.
|
||
|
my $hierarchy = "<synopsis>\n\n";
|
||
|
$level = 0;
|
||
|
for ($i = $#ancestors; $i >= 0; $i--) {
|
||
|
my $link_text;
|
||
|
# Don't add a link to the current widget, i.e. when i == 0.
|
||
|
if ($i > 0) {
|
||
|
my $ancestor_id = &CreateValidSGMLID ($ancestors[$i]);
|
||
|
$link_text = "<link linkend=\"$ancestor_id\">$ancestors[$i]</link>";
|
||
|
} else {
|
||
|
$link_text = "$ancestors[$i]";
|
||
|
}
|
||
|
if ($level == 0) {
|
||
|
$hierarchy .= " $link_text\n";
|
||
|
} else {
|
||
|
# $hierarchy .= ' ' x ($level * 6 - 3) . "|\n";
|
||
|
$hierarchy .= ' ' x ($level * 6 - 3) . "+----$link_text\n";
|
||
|
}
|
||
|
$level++;
|
||
|
}
|
||
|
$hierarchy .= "</synopsis>\n";
|
||
|
|
||
|
return <<EOF;
|
||
|
<refsect1>
|
||
|
<title>Object Hierarchy</title>
|
||
|
$hierarchy
|
||
|
</refsect1>
|
||
|
EOF
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : GetSignals
|
||
|
# Description : Returns the synopsis and detailed description DocBook output
|
||
|
# for the signal handlers of a given GtkObject subclass.
|
||
|
# Arguments : $object - the GtkObject subclass, e.g. 'GtkButton'.
|
||
|
#############################################################################
|
||
|
|
||
|
sub GetSignals {
|
||
|
my ($object) = @_;
|
||
|
my $synop = "";
|
||
|
my $desc = "";
|
||
|
|
||
|
my $i;
|
||
|
for ($i = 0; $i <= $#SignalObjects; $i++) {
|
||
|
if ($SignalObjects[$i] eq $object) {
|
||
|
# print "Found signal: $SignalNames[$i]\n";
|
||
|
my $name = $SignalNames[$i];
|
||
|
my $symbol = "${object}::${name}";
|
||
|
my $id = &CreateValidSGMLID ("$object-$name");
|
||
|
|
||
|
my $name_len = length ($name) + 2;
|
||
|
if ($name_len < $SIGNAL_FIELD_WIDTH) {
|
||
|
$synop .= ""<link linkend=\"$id\">$name</link>""
|
||
|
. (' ' x ($SIGNAL_FIELD_WIDTH - $name_len));
|
||
|
} else {
|
||
|
$synop .= ""<link linkend=\"$id\">$name</link>"\n"
|
||
|
. (' ' x $SIGNAL_FIELD_WIDTH);
|
||
|
}
|
||
|
|
||
|
$desc .= "<refsect2><title><anchor id=\"$id\">The "$name" signal</title>\n";
|
||
|
$desc .= "<programlisting>";
|
||
|
|
||
|
$SignalReturns[$i] =~ m/\s*(const\s*)?(\w+)\s*(\**)/;
|
||
|
my $type_modifier = defined($1) ? $1 : "";
|
||
|
my $type = $2;
|
||
|
my $pointer = $3;
|
||
|
my $xref = &MakeXRef ($type);
|
||
|
|
||
|
my $ret_type_len = length ($type_modifier) + length ($pointer)
|
||
|
+ length ($type);
|
||
|
my $ret_type_output = "$type_modifier$xref$pointer"
|
||
|
. (' ' x ($RETURN_TYPE_FIELD_WIDTH - $ret_type_len));
|
||
|
|
||
|
$synop .= "${ret_type_output}user_function (";
|
||
|
$desc .= "${ret_type_output}user_function (";
|
||
|
|
||
|
my @params = split ("\n", $SignalPrototypes[$i]);
|
||
|
my $j;
|
||
|
for ($j = 0; $j <= $#params; $j++) {
|
||
|
# allow alphanumerics, '_', '[' & ']' in param names
|
||
|
if ($params[$j] =~ m/^\s*(\w+)\s*(\**)\s*([\w\[\]]+)\s*$/) {
|
||
|
$type = $1;
|
||
|
$pointer = $2;
|
||
|
$name = $3;
|
||
|
$xref = &MakeXRef ($type);
|
||
|
$synop .= "$xref $pointer$name,\n";
|
||
|
$synop .= (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH));
|
||
|
$desc .= "$xref $pointer$name,\n";
|
||
|
$desc .= (' ' x ($SYMBOL_FIELD_WIDTH + $RETURN_TYPE_FIELD_WIDTH));
|
||
|
} else {
|
||
|
print "###Can't parse arg: $params[$j]\nArgs:$SignalPrototypes[$i]\n";
|
||
|
}
|
||
|
}
|
||
|
$xref = &MakeXRef ("gpointer");
|
||
|
$synop .= "$xref user_data);\n";
|
||
|
$desc .= "$xref user_data);</programlisting>\n";
|
||
|
|
||
|
if (defined ($SymbolDocs{$symbol})) {
|
||
|
$desc .= &ExpandAbbreviations($SymbolDocs{$symbol});
|
||
|
}
|
||
|
|
||
|
$desc .= &OutputParamDescriptions ("SIGNAL", $symbol);
|
||
|
$desc .= "</refsect2>";
|
||
|
}
|
||
|
}
|
||
|
if ($synop ne '') {
|
||
|
$synop = <<EOF;
|
||
|
<refsect1>
|
||
|
<title>Signal Prototypes</title>
|
||
|
<synopsis>
|
||
|
|
||
|
${synop}</synopsis>
|
||
|
</refsect1>
|
||
|
EOF
|
||
|
$desc = <<EOF;
|
||
|
<refsect1>
|
||
|
<title>Signals</title>
|
||
|
$desc
|
||
|
</refsect1>
|
||
|
EOF
|
||
|
}
|
||
|
return ($synop, $desc);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : GetArgs
|
||
|
# Description : Returns the synopsis and detailed description DocBook output
|
||
|
# for the Args of a given GtkObject subclass.
|
||
|
# Arguments : $object - the GtkObject subclass, e.g. 'GtkButton'.
|
||
|
#############################################################################
|
||
|
|
||
|
sub GetArgs {
|
||
|
my ($object) = @_;
|
||
|
my $synop = "";
|
||
|
my $desc = "";
|
||
|
|
||
|
my $i;
|
||
|
for ($i = 0; $i <= $#ArgObjects; $i++) {
|
||
|
if ($ArgObjects[$i] eq $object) {
|
||
|
# print "Found arg: $ArgNames[$i]\n";
|
||
|
my $name = $ArgNames[$i];
|
||
|
# Remember only one colon so we don't clash with signals.
|
||
|
my $symbol = "${object}:${name}";
|
||
|
# I've used two dashes here for the same reason.
|
||
|
my $id = &CreateValidSGMLID ("$object--$name");
|
||
|
|
||
|
my $type = $ArgTypes[$i];
|
||
|
my $type_output;
|
||
|
|
||
|
if ($type eq "GtkSignal") {
|
||
|
$type = "GtkSignalFunc, gpointer";
|
||
|
$type_output = &MakeXRef ("GtkSignalFunc") . ", "
|
||
|
. &MakeXRef ("gpointer");
|
||
|
} elsif ($type eq "GtkString") {
|
||
|
$type = "gchar*";
|
||
|
$type_output = &MakeXRef ("gchar") . "*";
|
||
|
} else {
|
||
|
$type_output = &MakeXRef ($type);
|
||
|
}
|
||
|
|
||
|
my $flags = $ArgFlags[$i];
|
||
|
my $flags_string = "";
|
||
|
|
||
|
if ($flags =~ m/r/) {
|
||
|
$flags_string = "Read";
|
||
|
}
|
||
|
if ($flags =~ m/w/) {
|
||
|
if ($flags_string) { $flags_string .= " / "; }
|
||
|
$flags_string .= "Write";
|
||
|
}
|
||
|
if ($flags =~ m/x/) {
|
||
|
if ($flags_string) { $flags_string .= " / "; }
|
||
|
$flags_string .= "Construct";
|
||
|
}
|
||
|
if ($flags =~ m/X/) {
|
||
|
if ($flags_string) { $flags_string .= " / "; }
|
||
|
$flags_string .= "Construct Only";
|
||
|
}
|
||
|
if ($flags =~ m/c/) {
|
||
|
if ($flags_string) { $flags_string .= " / "; }
|
||
|
$flags_string .= "Child";
|
||
|
}
|
||
|
|
||
|
my $pad1 = " " x (20 - length ($name));
|
||
|
my $pad2 = " " x (20 - length ($type));
|
||
|
$synop .= " "<link linkend=\"$id\">$name</link>"$pad1 $type_output$pad2 : $flags_string\n";
|
||
|
|
||
|
$desc .= "<varlistentry><term><anchor id=\"$id\">"<literal>$name</literal>" ($type_output : $flags_string)</term>\n<listitem>\n";
|
||
|
|
||
|
if (defined ($SymbolDocs{$symbol})) {
|
||
|
$desc .= &ExpandAbbreviations($SymbolDocs{$symbol});
|
||
|
} else {
|
||
|
$desc .= "<para></para>\n";
|
||
|
}
|
||
|
|
||
|
$desc .= "</listitem></varlistentry>\n";
|
||
|
}
|
||
|
}
|
||
|
if ($synop ne '') {
|
||
|
$synop = <<EOF;
|
||
|
<refsect1>
|
||
|
<title>Args</title>
|
||
|
<synopsis>
|
||
|
|
||
|
${synop}</synopsis>
|
||
|
</refsect1>
|
||
|
EOF
|
||
|
$desc = <<EOF;
|
||
|
<refsect1>
|
||
|
<title>Args</title>
|
||
|
<variablelist>
|
||
|
$desc
|
||
|
</variablelist>
|
||
|
</refsect1>
|
||
|
EOF
|
||
|
}
|
||
|
return ($synop, $desc);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ReadSourceDocumentation
|
||
|
# Description : This reads in the documentation embedded in comment blocks
|
||
|
# in the source code (for Gnome).
|
||
|
#
|
||
|
# Parameter descriptions override any in the template files.
|
||
|
# Function descriptions are placed before any description from
|
||
|
# the template files.
|
||
|
#
|
||
|
# It recursively descends the source directory looking for .c
|
||
|
# files and scans them looking for specially-formatted comment
|
||
|
# blocks.
|
||
|
#
|
||
|
# Arguments : $source_dir - the directory to scan.
|
||
|
#############################################################################
|
||
|
|
||
|
sub ReadSourceDocumentation {
|
||
|
my ($source_dir) = @_;
|
||
|
# print "Scanning source directory: $source_dir\n";
|
||
|
|
||
|
# This array holds any subdirectories found.
|
||
|
my (@subdirs) = ();
|
||
|
|
||
|
opendir (SRCDIR, $source_dir)
|
||
|
|| die "Can't open source directory $source_dir: $!";
|
||
|
my $file;
|
||
|
foreach $file (readdir (SRCDIR)) {
|
||
|
if ($file =~ /^\./) {
|
||
|
next;
|
||
|
} elsif (-d "$source_dir/$file") {
|
||
|
push (@subdirs, $file);
|
||
|
} elsif ($file =~ m/\.c$/) {
|
||
|
&ScanSourceFile ("$source_dir/$file");
|
||
|
}
|
||
|
}
|
||
|
closedir (SRCDIR);
|
||
|
|
||
|
# Now recursively scan the subdirectories.
|
||
|
my $dir;
|
||
|
foreach $dir (@subdirs) {
|
||
|
&ReadSourceDocumentation ("$source_dir/$dir");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ScanSourceFile
|
||
|
# Description : Scans one source file looking for specially-formatted comment
|
||
|
# blocks. It calls &MergeSourceDocumentation to merge any
|
||
|
# documentation found with the documentation already read in
|
||
|
# from the template files.
|
||
|
#
|
||
|
# Arguments : $file - the file to scan.
|
||
|
#############################################################################
|
||
|
|
||
|
sub ScanSourceFile {
|
||
|
my ($file) = @_;
|
||
|
|
||
|
# print "Scanning source file: $file\n";
|
||
|
|
||
|
open (SRCFILE, $file)
|
||
|
|| die "Can't open $file: $!";
|
||
|
my $in_comment_block = 0;
|
||
|
my $symbol;
|
||
|
my ($in_description, $in_return);
|
||
|
my ($description, $return_desc, $return_start);
|
||
|
my $current_param;
|
||
|
my @params;
|
||
|
while (<SRCFILE>) {
|
||
|
# Look for the start of a comment block.
|
||
|
if (!$in_comment_block) {
|
||
|
if (m%^\s*/\*\*\s%) {
|
||
|
# print "Found comment block start\n";
|
||
|
$in_comment_block = 1;
|
||
|
|
||
|
# Reset all the symbol data.
|
||
|
$symbol = "";
|
||
|
$in_description = 0;
|
||
|
$in_return = 0;
|
||
|
$description = "";
|
||
|
$return_desc = "";
|
||
|
$current_param = -1;
|
||
|
@params = ();
|
||
|
}
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
# We're in a comment block. Check if we've found the end of it.
|
||
|
if (m%^\s*\*+/%) {
|
||
|
# print "Found comment block end: $symbol\n";
|
||
|
if (!$symbol) {
|
||
|
print <<EOF;
|
||
|
WARNING: symbol name not found in comment block.
|
||
|
$file line $.
|
||
|
EOF
|
||
|
} else {
|
||
|
# Add the return value description onto the end of the params.
|
||
|
if ($return_desc) {
|
||
|
push (@params, "Returns");
|
||
|
push (@params, $return_desc);
|
||
|
}
|
||
|
$SourceSymbolDocs{$symbol} = $description;
|
||
|
$SourceSymbolParams{$symbol} = [ @params ];
|
||
|
}
|
||
|
|
||
|
$in_comment_block = 0;
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
# Get rid of ' * ' at start of every line in the comment block.
|
||
|
s%^\s*\*\s*%%;
|
||
|
# But make sure we don't get rid of the newline at the end.
|
||
|
if (!$_) {
|
||
|
$_ = "\n";
|
||
|
}
|
||
|
|
||
|
# If we haven't found the symbol name yet, look for it.
|
||
|
if (!$symbol) {
|
||
|
if (m%^(\w+)\s*:?%) {
|
||
|
$symbol = $1;
|
||
|
}
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
# If we're in the return value description, add it to the end.
|
||
|
if ($in_return) {
|
||
|
# If we find another valid returns line, we assume that the first
|
||
|
# one was really part of the description.
|
||
|
if (m%^(returns:|return\s+value:|returns\s*)%i) {
|
||
|
$description .= $return_start . $return_desc;
|
||
|
$return_start = $1;
|
||
|
$return_desc = $';
|
||
|
} else {
|
||
|
$return_desc .= $_;
|
||
|
}
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
# If we're in the description part, check for the 'Return' line.
|
||
|
# If that isn't found, add the text to the end.
|
||
|
if ($in_description) {
|
||
|
# Get rid of 'Description:'
|
||
|
s%^Description:%%;
|
||
|
|
||
|
if (m%^(returns:|return\s+value:|returns\s*)%i) {
|
||
|
# print "RETURNS: $_";
|
||
|
$return_start = $1;
|
||
|
$return_desc = $';
|
||
|
$in_return = 1;
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
$description .= $_;
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
# We must be in the parameters. Check for the empty line at the end.
|
||
|
if (m%^$%) {
|
||
|
$in_description = 1;
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
# Look for a parameter name.
|
||
|
if (m%^@(\S+)\s*:%) {
|
||
|
my $param_name = $1;
|
||
|
# print "Found parameter: $param_name\n";
|
||
|
# Allow '...' as the Varargs parameter.
|
||
|
if ($param_name eq "...") {
|
||
|
$param_name = "Varargs";
|
||
|
}
|
||
|
push (@params, $param_name);
|
||
|
push (@params, $');
|
||
|
$current_param += 2;
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
# We must be in the middle of a parameter description, so add it on
|
||
|
# to the last element in @params.
|
||
|
if ($current_param == -1) {
|
||
|
print <<EOF
|
||
|
ERROR parsing comment block file : parameter expected -
|
||
|
$file:$.
|
||
|
EOF
|
||
|
} else {
|
||
|
$params[$#params] .= $_;
|
||
|
}
|
||
|
}
|
||
|
close (SRCFILE);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : MergeSourceDocumentation
|
||
|
# Description : This merges documentation read from a source file into the
|
||
|
# documentation read in from a template file.
|
||
|
#
|
||
|
# Parameter descriptions override any in the template files.
|
||
|
# Function descriptions are placed before any description from
|
||
|
# the template files.
|
||
|
#
|
||
|
# Arguments : none
|
||
|
#############################################################################
|
||
|
|
||
|
sub MergeSourceDocumentation {
|
||
|
my $symbol;
|
||
|
foreach $symbol (keys (%SymbolDocs)) {
|
||
|
if (exists ($SourceSymbolDocs{$symbol})) {
|
||
|
my $src_doc = $SourceSymbolDocs{$symbol};
|
||
|
my $tmpl_doc = $SymbolDocs{$symbol};
|
||
|
$tmpl_doc = defined ($tmpl_doc) ? $tmpl_doc : "";
|
||
|
$src_doc =~ s/^\s+//;
|
||
|
$src_doc =~ s/\s+$//;
|
||
|
|
||
|
# Convert special SGML characters. I'm not sure if we want to do
|
||
|
# this but currently there are a couple of '&'s in the source code
|
||
|
# comment blocks which mess up the HTML output badly.
|
||
|
$src_doc = &CreateValidSGML ($src_doc);
|
||
|
|
||
|
# If there is a blank line, finish the paragraph and start another.
|
||
|
if ($src_doc =~ s%\n{2,}%\n</para>\n<para>\n%g) {
|
||
|
# print "Converted blank lines:\n$src_doc\n";
|
||
|
}
|
||
|
$SymbolDocs{$symbol} = "<para>\n$src_doc</para>\n$tmpl_doc";
|
||
|
|
||
|
# The templates contain the definitive parameter names and order,
|
||
|
# so we will not change that. We only override the actual text.
|
||
|
my $tmpl_params = $SymbolParams{$symbol};
|
||
|
if (!defined ($tmpl_params)) {
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
my $params = $SourceSymbolParams{$symbol};
|
||
|
my $j;
|
||
|
for ($j = 0; $j <= $#$tmpl_params; $j += 2) {
|
||
|
my $tmpl_param_name = $$tmpl_params[$j];
|
||
|
my $tmpl_param_desc = $$tmpl_params[$j + 1];
|
||
|
|
||
|
# Try to find the param in the source comment documentation.
|
||
|
my $found = 0;
|
||
|
my $k;
|
||
|
for ($k = 0; $k <= $#$params; $k += 2) {
|
||
|
my $param_name = $$params[$k];
|
||
|
my $param_desc = $$params[$k + 1];
|
||
|
|
||
|
# We accept changed in case, since the Gnome source docs
|
||
|
# contain a lot of these.
|
||
|
if ("\L$param_name" eq "\L$tmpl_param_name") {
|
||
|
$found = 1;
|
||
|
|
||
|
# Override the description.
|
||
|
$$tmpl_params[$j + 1] = &CreateValidSGML ($param_desc);
|
||
|
|
||
|
# Set the name to "" to mark it as used.
|
||
|
$$params[$k] = "";
|
||
|
last;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
# Output a warning if the parameter is not found.
|
||
|
if (!$found) {
|
||
|
print <<EOF;
|
||
|
WARNING: Parameter description missing in source code comment block -
|
||
|
Func: $symbol Param: $tmpl_param_name.
|
||
|
EOF
|
||
|
}
|
||
|
}
|
||
|
|
||
|
# Now we output a warning if parameters have been described which
|
||
|
# do not exist.
|
||
|
for ($j = 0; $j <= $#$params; $j += 2) {
|
||
|
my $param_name = $$params[$j];
|
||
|
if ($param_name) {
|
||
|
print <<EOF;
|
||
|
WARNING: Parameter described in source code comment block but does not exist -
|
||
|
Func: $symbol Param: $param_name.
|
||
|
EOF
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# LIBRARY FUNCTIONS - These functions are used in both gtkdoc-mkdb and
|
||
|
# gtkdoc-mktmpl and should eventually be moved to a
|
||
|
# separate library.
|
||
|
#############################################################################
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ReadDeclarationsFile
|
||
|
# Description : This reads in a file containing the function/macro/enum etc.
|
||
|
# declarations.
|
||
|
#
|
||
|
# Note that in some cases there are several declarations with
|
||
|
# the same name, e.g. for conditional macros. In this case we
|
||
|
# set a flag in the %DeclarationConditional hash so the
|
||
|
# declaration is not shown in the docs.
|
||
|
#
|
||
|
# If a macro and a function have the same name, e.g. for
|
||
|
# gtk_object_ref, the function declaration takes precedence.
|
||
|
#
|
||
|
# Some opaque structs are just declared with 'typedef struct
|
||
|
# _name name;' in which case the declaration may be empty.
|
||
|
# The structure may have been found later in the header, so
|
||
|
# that overrides the empty declaration.
|
||
|
#
|
||
|
# Arguments : $file - the declarations file to read
|
||
|
# $override - if declarations in this file should override
|
||
|
# any current declaration.
|
||
|
#############################################################################
|
||
|
|
||
|
sub ReadDeclarationsFile {
|
||
|
my ($file, $override) = @_;
|
||
|
|
||
|
if ($override == 0) {
|
||
|
%Declarations = ();
|
||
|
%DeclarationTypes = ();
|
||
|
%DeclarationConditional = ();
|
||
|
%DeclarationOutput = ();
|
||
|
}
|
||
|
|
||
|
open (INPUT, $file)
|
||
|
|| die "Can't open $file";
|
||
|
my $declaration_type = "";
|
||
|
my $declaration_name;
|
||
|
my $declaration;
|
||
|
while (<INPUT>) {
|
||
|
if (!$declaration_type) {
|
||
|
if (m/^<([^>]+)>/) {
|
||
|
$declaration_type = $1;
|
||
|
$declaration_name = "";
|
||
|
# print "Found declaration: $declaration_type\n";
|
||
|
$declaration = "";
|
||
|
}
|
||
|
} else {
|
||
|
if (m%^<NAME>(.*)</NAME>%) {
|
||
|
$declaration_name = $1;
|
||
|
} elsif (m%^</$declaration_type>%) {
|
||
|
# print "Found end of declaration: $declaration_name\n";
|
||
|
# Check that the declaration has a name
|
||
|
if ($declaration_name eq "") {
|
||
|
print "ERROR: $declaration_type has no name $file:$.\n";
|
||
|
}
|
||
|
|
||
|
# Check if the symbol is already defined.
|
||
|
if (defined ($Declarations{$declaration_name})
|
||
|
&& $override == 0) {
|
||
|
# Function declarations take precedence.
|
||
|
if ($DeclarationTypes{$declaration_name} eq 'FUNCTION') {
|
||
|
# Ignore it.
|
||
|
} elsif ($declaration_type eq 'FUNCTION') {
|
||
|
$Declarations{$declaration_name} = $declaration;
|
||
|
$DeclarationTypes{$declaration_name} = $declaration_type;
|
||
|
} elsif ($DeclarationTypes{$declaration_name}
|
||
|
eq $declaration_type) {
|
||
|
# If the existing declaration is empty override it.
|
||
|
if ($declaration_type eq 'STRUCT') {
|
||
|
if ($Declarations{$declaration_name} =~ m/^\s*$/) {
|
||
|
$Declarations{$declaration_name} = $declaration;
|
||
|
} elsif ($declaration =~ m/^\s*$/) {
|
||
|
# Ignore an empty declaration.
|
||
|
} else {
|
||
|
print "WARNING: Structure has multiple definitions: $declaration_name\n";
|
||
|
}
|
||
|
|
||
|
} else {
|
||
|
# set flag in %DeclarationConditional hash for
|
||
|
# multiply defined macros/typedefs.
|
||
|
$DeclarationConditional{$declaration_name} = 1;
|
||
|
}
|
||
|
} else {
|
||
|
print "ERROR: $declaration_name has multiple definitions\n";
|
||
|
}
|
||
|
} else {
|
||
|
$Declarations{$declaration_name} = $declaration;
|
||
|
$DeclarationTypes{$declaration_name} = $declaration_type;
|
||
|
}
|
||
|
$declaration_type = "";
|
||
|
} else {
|
||
|
$declaration .= $_;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
close (INPUT);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ReadSignalsFile
|
||
|
# Description : This reads in an existing file which contains information on
|
||
|
# all GTK signals. It creates the arrays @SignalNames and
|
||
|
# @SignalPrototypes containing info on the signals. The first
|
||
|
# line of the SignalPrototype is the return type of the signal
|
||
|
# handler. The remaining lines are the parameters passed to it.
|
||
|
# The last parameter, "gpointer user_data" is always the same
|
||
|
# so is not included.
|
||
|
# Arguments : $file - the file containing the signal handler prototype
|
||
|
# information.
|
||
|
#############################################################################
|
||
|
|
||
|
sub ReadSignalsFile {
|
||
|
my ($file) = @_;
|
||
|
|
||
|
my $in_signal = 0;
|
||
|
my $signal_object;
|
||
|
my $signal_name;
|
||
|
my $signal_returns;
|
||
|
my $signal_prototype;
|
||
|
|
||
|
# Reset the signal info.
|
||
|
@SignalObjects = ();
|
||
|
@SignalNames = ();
|
||
|
@SignalReturns = ();
|
||
|
@SignalPrototypes = ();
|
||
|
|
||
|
if (! -f $file) {
|
||
|
return;
|
||
|
}
|
||
|
if (!open (INPUT, $file)) {
|
||
|
warn "Can't open $file - skipping signals\n";
|
||
|
return;
|
||
|
}
|
||
|
while (<INPUT>) {
|
||
|
if (!$in_signal) {
|
||
|
if (m/^<SIGNAL>/) {
|
||
|
$in_signal = 1;
|
||
|
$signal_object = "";
|
||
|
$signal_name = "";
|
||
|
$signal_returns = "";
|
||
|
$signal_prototype = "";
|
||
|
}
|
||
|
} else {
|
||
|
if (m/^<NAME>(.*)<\/NAME>/) {
|
||
|
$signal_name = $1;
|
||
|
if ($signal_name =~ m/^(.*)::(.*)$/) {
|
||
|
$signal_object = $1;
|
||
|
$signal_name = $2;
|
||
|
# print "Found signal: $signal_name\n";
|
||
|
} else {
|
||
|
print "Invalid signal name: $signal_name\n";
|
||
|
}
|
||
|
} elsif (m/^<RETURNS>(.*)<\/RETURNS>/) {
|
||
|
$signal_returns = $1;
|
||
|
} elsif (m%^</SIGNAL>%) {
|
||
|
# print "Found end of signal: ${signal_object}::${signal_name}\nReturns: ${signal_returns}\n${signal_prototype}";
|
||
|
push (@SignalObjects, $signal_object);
|
||
|
push (@SignalNames, $signal_name);
|
||
|
push (@SignalReturns, $signal_returns);
|
||
|
push (@SignalPrototypes, $signal_prototype);
|
||
|
$in_signal = 0;
|
||
|
} else {
|
||
|
$signal_prototype .= $_;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
close (INPUT);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ReadTemplateFile
|
||
|
# Description : This reads in the manually-edited documentation file
|
||
|
# corresponding to the file currently being created, so we can
|
||
|
# insert the documentation at the appropriate places.
|
||
|
# It outputs %SymbolTypes, %SymbolDocs and %SymbolParams, which
|
||
|
# is a hash of arrays.
|
||
|
# NOTE: This function is duplicated in gtkdoc-mkdb (but
|
||
|
# slightly different).
|
||
|
# Arguments : $docsfile - the template file to read in.
|
||
|
# $skip_unused_params - 1 if the unused parameters should be
|
||
|
# skipped.
|
||
|
#############################################################################
|
||
|
|
||
|
sub ReadTemplateFile {
|
||
|
my ($docsfile, $skip_unused_params) = @_;
|
||
|
|
||
|
# print "Reading $docsfile\n";
|
||
|
if (! -f $docsfile) {
|
||
|
print "File doesn't exist: $docsfile\n";
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
my $current_type = ""; # Type of symbol being read.
|
||
|
my $current_symbol = ""; # Name of symbol being read.
|
||
|
my $symbol_doc = ""; # Description of symbol being read.
|
||
|
my @params; # Parameter names and descriptions of current
|
||
|
# function/macro/function typedef.
|
||
|
my $current_param = -1; # Index of parameter currently being read.
|
||
|
# Note that the param array contains pairs
|
||
|
# of param name & description.
|
||
|
my $in_unused_params = 0; # True if we are reading in the unused params.
|
||
|
|
||
|
open (DOCS, $docsfile)
|
||
|
|| die "Can't open file $docsfile: $!";
|
||
|
while (<DOCS>) {
|
||
|
if (m/^<!-- ##### ([A-Z_]+) (\S+) ##### -->/) {
|
||
|
my $type = $1;
|
||
|
my $symbol = $2;
|
||
|
if ($symbol eq "Title"
|
||
|
|| $symbol eq "Short_Description"
|
||
|
|| $symbol eq "Long_Description"
|
||
|
|| $symbol eq "See_Also") {
|
||
|
$symbol = $docsfile . ":" . $symbol;
|
||
|
# print "Found symbol: $symbol\n";
|
||
|
}
|
||
|
|
||
|
# Store previous symbol, but remove any trailing blank lines.
|
||
|
if ($current_symbol ne "") {
|
||
|
$symbol_doc =~ s/\s+$//;
|
||
|
$SymbolTypes{$current_symbol} = $current_type;
|
||
|
$SymbolDocs{$current_symbol} = $symbol_doc;
|
||
|
if ($current_param >= 0) {
|
||
|
$SymbolParams{$current_symbol} = [ @params ];
|
||
|
} else {
|
||
|
# Delete any existing params in case we are overriding a
|
||
|
# previously read template.
|
||
|
delete $SymbolParams{$current_symbol};
|
||
|
}
|
||
|
}
|
||
|
$current_type = $type;
|
||
|
$current_symbol = $symbol;
|
||
|
$current_param = -1;
|
||
|
$in_unused_params = 0;
|
||
|
$symbol_doc = "";
|
||
|
@params = ();
|
||
|
|
||
|
} elsif (m/^<!-- # Unused Parameters # -->/) {
|
||
|
# print "DEBUG: Found unused parameters\n";
|
||
|
$in_unused_params = 1;
|
||
|
next;
|
||
|
|
||
|
} elsif ($in_unused_params && $skip_unused_params) {
|
||
|
# When outputting the DocBook we skip unused parameters.
|
||
|
# print "DEBUG: Skipping unused param: $_";
|
||
|
next;
|
||
|
|
||
|
} else {
|
||
|
# Check if param found
|
||
|
if (s/^\@(\S+):\s*//) {
|
||
|
my $param_name = $1;
|
||
|
# Allow variations of 'Returns'
|
||
|
if ($param_name =~ m/^[Rr]eturns?$/) {
|
||
|
$param_name = "Returns";
|
||
|
}
|
||
|
# print "Found param: $param_name\n";
|
||
|
push (@params, $param_name);
|
||
|
push (@params, $_);
|
||
|
$current_param += 2;
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
if ($current_param >= 0) {
|
||
|
$params[$current_param] .= $_;
|
||
|
} else {
|
||
|
$symbol_doc .= $_;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
# Remember to finish the current symbol doccs.
|
||
|
if ($current_symbol ne "") {
|
||
|
$symbol_doc =~ s/\s+$//;
|
||
|
$SymbolTypes{$current_symbol} = $current_type;
|
||
|
$SymbolDocs{$current_symbol} = $symbol_doc;
|
||
|
if ($current_param >= 0) {
|
||
|
$SymbolParams{$current_symbol} = [ @params ];
|
||
|
} else {
|
||
|
delete $SymbolParams{$current_symbol};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
close (DOCS);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ReadObjectHierarchy
|
||
|
# Description : This reads in the $MODULE-hierarchy.txt file containing all
|
||
|
# the GtkObject subclasses described in this module (and their
|
||
|
# ancestors).
|
||
|
# It places them in the @Objects array, and places their level
|
||
|
# in the widget hierarchy in the @ObjectLevels array, at the
|
||
|
# same index. GtkObject, the root object, has a level of 1.
|
||
|
#
|
||
|
# FIXME: the version in gtkdoc-mkdb also generates tree_index.sgml
|
||
|
# as it goes along, this should be split out into a separate
|
||
|
# function.
|
||
|
#
|
||
|
# Arguments : none
|
||
|
#############################################################################
|
||
|
|
||
|
sub ReadObjectHierarchy {
|
||
|
@Objects = ();
|
||
|
@ObjectLevels = ();
|
||
|
|
||
|
if (! -f $OBJECT_TREE_FILE) {
|
||
|
return;
|
||
|
}
|
||
|
if (!open (INPUT, $OBJECT_TREE_FILE)) {
|
||
|
warn "Can't open $OBJECT_TREE_FILE - skipping object tree\n";
|
||
|
return;
|
||
|
}
|
||
|
open (OUTPUT, ">$SGML_OUTPUT_DIR/tree_index.sgml")
|
||
|
|| die "Can't create $SGML_OUTPUT_DIR/tree_index.sgml";
|
||
|
print (OUTPUT "<literallayout>\n");
|
||
|
|
||
|
while (<INPUT>) {
|
||
|
if (m/\S+/) {
|
||
|
my $object = $&;
|
||
|
my $level = (length($`)) / 2 + 1;
|
||
|
# print ("Level: $level Object: $object\n");
|
||
|
|
||
|
my $xref = &MakeXRef ($object);
|
||
|
print (OUTPUT ' ' x ($level * 4), "$xref\n");
|
||
|
push (@Objects, $object);
|
||
|
push (@ObjectLevels, $level);
|
||
|
}
|
||
|
}
|
||
|
print (OUTPUT "</literallayout>\n");
|
||
|
|
||
|
close (INPUT);
|
||
|
close (OUTPUT);
|
||
|
|
||
|
&OutputObjectList;
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ReadArgsFile
|
||
|
# Description : This reads in an existing file which contains information on
|
||
|
# all GTK args. It creates the arrays @ArgObjects, @ArgNames,
|
||
|
# @ArgTypes and @ArgFlags containing info on the args.
|
||
|
# Arguments : $file - the file containing the arg information.
|
||
|
#############################################################################
|
||
|
|
||
|
sub ReadArgsFile {
|
||
|
my ($file) = @_;
|
||
|
|
||
|
my $in_arg = 0;
|
||
|
my $arg_object;
|
||
|
my $arg_name;
|
||
|
my $arg_type;
|
||
|
my $arg_flags;
|
||
|
|
||
|
# Reset the signal info.
|
||
|
@ArgObjects = ();
|
||
|
@ArgNames = ();
|
||
|
@ArgTypes = ();
|
||
|
@ArgFlags = ();
|
||
|
|
||
|
if (! -f $file) {
|
||
|
return;
|
||
|
}
|
||
|
if (!open (INPUT, $file)) {
|
||
|
warn "Can't open $file - skipping args\n";
|
||
|
return;
|
||
|
}
|
||
|
while (<INPUT>) {
|
||
|
if (!$in_arg) {
|
||
|
if (m/^<ARG>/) {
|
||
|
$in_arg = 1;
|
||
|
$arg_object = "";
|
||
|
$arg_name = "";
|
||
|
$arg_type = "";
|
||
|
$arg_flags = "";
|
||
|
}
|
||
|
} else {
|
||
|
if (m/^<NAME>(.*)<\/NAME>/) {
|
||
|
$arg_name = $1;
|
||
|
if ($arg_name =~ m/^(.*)::(.*)$/) {
|
||
|
$arg_object = $1;
|
||
|
$arg_name = $2;
|
||
|
# print "Found arg: $arg_name\n";
|
||
|
} else {
|
||
|
print "Invalid arg name: $arg_name\n";
|
||
|
}
|
||
|
} elsif (m/^<TYPE>(.*)<\/TYPE>/) {
|
||
|
$arg_type = $1;
|
||
|
} elsif (m/^<FLAGS>(.*)<\/FLAGS>/) {
|
||
|
$arg_flags = $1;
|
||
|
} elsif (m%^</ARG>%) {
|
||
|
# print "Found end of arg: ${arg_object}::${arg_name}\n${arg_type} : ${arg_flags}\n";
|
||
|
push (@ArgObjects, $arg_object);
|
||
|
push (@ArgNames, $arg_name);
|
||
|
push (@ArgTypes, $arg_type);
|
||
|
push (@ArgFlags, $arg_flags);
|
||
|
$in_arg = 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
close (INPUT);
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : CheckIsObject
|
||
|
# Description : Returns 1 if the given name is a GtkObject or a subclass.
|
||
|
# It uses the global @Objects array.
|
||
|
# Note that the @Objects array only contains classes in the
|
||
|
# current module and their ancestors - not all GTK classes.
|
||
|
# Arguments : $name - the name to check.
|
||
|
#############################################################################
|
||
|
|
||
|
sub CheckIsObject {
|
||
|
my ($name) = @_;
|
||
|
|
||
|
my $object;
|
||
|
foreach $object (@Objects) {
|
||
|
if ($object eq $name) {
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ParseStructDeclaration
|
||
|
# Description : This function takes a structure declaration and
|
||
|
# breaks it into individual type declarations.
|
||
|
# Arguments : $declaration - the declaration to parse
|
||
|
# $is_object - true if this is an object structure
|
||
|
# $typefunc - function reference to apply to type
|
||
|
# $namefunc - function reference to apply to name
|
||
|
#############################################################################
|
||
|
|
||
|
sub ParseStructDeclaration {
|
||
|
my ($declaration, $is_object, $typefunc, $namefunc) = @_;
|
||
|
|
||
|
# Remove all private parts of the declaration
|
||
|
|
||
|
# For objects, assume private
|
||
|
if ($is_object) {
|
||
|
$declaration =~ s!(struct\s+\w*\s*\{)
|
||
|
.*?
|
||
|
(?:/\*\s*<\s*public\s*>\s*\*/|(?=\}))!$1!msgx;
|
||
|
}
|
||
|
|
||
|
$declaration =~ s!\n?[ \t]*/\*\s*<\s*private\s*>\s*\*/
|
||
|
.*?
|
||
|
(?:/\*\s*<\s*public\s*>\s*\*/|(?=\}))!!msgx;
|
||
|
|
||
|
# Remove all other comments;
|
||
|
$declaration =~ s@/\*([^*]+|\*(?!/))*\*/@ @g;
|
||
|
|
||
|
my @result = ();
|
||
|
|
||
|
if ($declaration =~ /^\s*$/) {
|
||
|
return @result;
|
||
|
}
|
||
|
|
||
|
# Prime match after "struct {" declaration
|
||
|
if (!scalar($declaration =~ m/struct\s+\w*\s*\{/msg)) {
|
||
|
die "Structure declaration '$declaration' does not begin with struct [NAME] {\n";
|
||
|
}
|
||
|
|
||
|
# Treat lines in sequence, allowing singly nested anonymous structs
|
||
|
# and unions.
|
||
|
while ($declaration =~ m/\s*([^{;]+(\{[^\}]*\}[^{;]+)?);/msg) {
|
||
|
my $line = $1;
|
||
|
|
||
|
last if $line =~ /^\s*\}\s*\w*\s*$/;
|
||
|
|
||
|
# FIXME: Just ignore nested structs and unions for now
|
||
|
next if $line =~ /{/;
|
||
|
|
||
|
# FIXME: The regexes here are the same as in OutputFunction;
|
||
|
# this functionality should be separated out.
|
||
|
|
||
|
if ($line =~ m/^
|
||
|
(const\s+|unsigned\s+)*(struct\s+)? # mod1
|
||
|
(\w+)\s* # type
|
||
|
(\**)\s* # ptr1
|
||
|
(const\s+)? # mod2
|
||
|
(\**)?\s* # ptr2
|
||
|
(\w+(?:\s*,\s*\w+)*)\s* # name
|
||
|
(?:((?:\[[^\]]*\]\s*)+) | # array
|
||
|
(:\s*\d+))?\s* # bits
|
||
|
$/x) {
|
||
|
my $mod1 = defined($1) ? $1 : "";
|
||
|
if (defined($2)) { $mod1 .= $2; }
|
||
|
my $type = $3;
|
||
|
my $ptr1 = $4;
|
||
|
my $mod2 = defined($5) ? $5 : "";
|
||
|
my $ptr2 = $6;
|
||
|
my $name = $7;
|
||
|
$ptr1 = " " . $ptr1;
|
||
|
my $array = defined($8) ? $8 : "";
|
||
|
my $bits = defined($9) ? " $9" : "";
|
||
|
my $ptype = defined $typefunc ? $typefunc->($type) : $type;
|
||
|
|
||
|
# FIXME:
|
||
|
# As a hack, we allow the "name" to be of the form
|
||
|
# "a, b, c". This isn't the correct C syntax, but
|
||
|
# at least we get "gint16 x, y" right. Such constructs
|
||
|
# should really be completely removed from the source.
|
||
|
# Or we should really try to understand the C syntax
|
||
|
# here...
|
||
|
|
||
|
my @names = split /\s*,\s*/, $name;
|
||
|
for my $n (@names) {
|
||
|
push @result, $n;
|
||
|
if (defined $namefunc) {
|
||
|
$n = $namefunc->($n);
|
||
|
}
|
||
|
push @result, "$mod1$ptype$ptr1$mod2$ptr2$n$array$bits";
|
||
|
}
|
||
|
|
||
|
# Try to match structure members which are functions
|
||
|
} elsif ($line =~ m/^
|
||
|
(const\s+|unsigned\s+)*(struct\s+)? # mod1
|
||
|
(\w+)\s* # type
|
||
|
(\**)\s* # ptr1
|
||
|
(const\s+)? # mod2
|
||
|
\(\s*\*\s*(\w+)\s*\)\s* # name
|
||
|
\(([^)]*)\)\s* # func_params
|
||
|
$/x) {
|
||
|
|
||
|
my $mod1 = defined($1) ? $1 : "";
|
||
|
if (defined($2)) { $mod1 .= $2; }
|
||
|
my $type = $3;
|
||
|
my $ptr1 = $4;
|
||
|
my $mod2 = defined($5) ? $5 : "";
|
||
|
my $name = $6;
|
||
|
my $func_params = $7;
|
||
|
my $ptype = defined $typefunc ? $typefunc->($type) : $type;
|
||
|
my $pname = defined $namefunc ? $namefunc->($name) : $name;
|
||
|
|
||
|
push @result, $name;
|
||
|
push @result, "$mod1$ptype$ptr1$mod2 (*$pname) ($func_params)";
|
||
|
|
||
|
} else {
|
||
|
warn "Cannot parse structure field $line";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return @result;
|
||
|
}
|
||
|
|
||
|
|
||
|
#############################################################################
|
||
|
# Function : ParseEnumDeclaration
|
||
|
# Description : This function takes a enumeration declaration and
|
||
|
# breaks it into individual enum member declarations.
|
||
|
# Arguments : $declaration - the declaration to parse
|
||
|
#############################################################################
|
||
|
|
||
|
sub ParseEnumDeclaration {
|
||
|
my ($declaration, $is_object) = @_;
|
||
|
|
||
|
# Remove comments;
|
||
|
$declaration =~ s@/\*([^*]+|\*(?!/))*\*/@ @g;
|
||
|
|
||
|
my @result = ();
|
||
|
|
||
|
if ($declaration =~ /^\s*$/) {
|
||
|
return @result;
|
||
|
}
|
||
|
|
||
|
# Remove parenthesized expressions (in macros like GTK_BLAH = BLAH(1,3))
|
||
|
# to avoid getting confused by commas they might contain. This
|
||
|
# doesn't handle nested parentheses correctly.
|
||
|
|
||
|
$declaration =~ s/\([^)]+\)//g;
|
||
|
|
||
|
# Prime match after "typedef enum {" declaration
|
||
|
if (!scalar($declaration =~ m/typedef\s+enum\s*\{/msg)) {
|
||
|
die "Enum declaration '$declaration' does not begin with typedef enum {\n";
|
||
|
}
|
||
|
|
||
|
# Treat lines in sequence.
|
||
|
while ($declaration =~ m/\s*([^,\}]+)([,\}])/msg) {
|
||
|
my $line = $1;
|
||
|
my $terminator = $2;
|
||
|
|
||
|
if ($line =~ m/^(\w+)\s*(=.*)?$/msg) {
|
||
|
push @result, $1;
|
||
|
|
||
|
# Special case for GIOCondition, where the values are specified by
|
||
|
# macros which expand to include the equal sign like '=1'.
|
||
|
} elsif ($line =~ m/^(\w+)\s*GLIB_SYSDEF_POLL/msg) {
|
||
|
push @result, $1;
|
||
|
|
||
|
# Special case include of <gdk/gdkcursors.h>, just ignore it
|
||
|
} elsif ($line =~ m/^#include/) {
|
||
|
last;
|
||
|
|
||
|
} else {
|
||
|
warn "Cannot parse enumeration member $line";
|
||
|
}
|
||
|
|
||
|
last if $terminator eq '}';
|
||
|
}
|
||
|
|
||
|
return @result;
|
||
|
}
|