1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: Parser of the API/interface XML files
4 // Author: Francesco Montorsi
7 // Copyright: (c) 2008 Francesco Montorsi
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
11 // For compilers that support precompilation, includes "wx/wx.h".
12 #include "wx/wxprec.h"
18 // for all others, include the necessary headers
23 #include "wx/xml/xml.h"
24 #include "wx/wfstream.h"
25 #include "wx/hashmap.h"
26 #include "wx/filename.h"
27 #include "xmlparser.h"
30 #include <wx/arrimpl.cpp>
31 WX_DEFINE_OBJARRAY(wxTypeArray
)
32 WX_DEFINE_OBJARRAY(wxArgumentTypeArray
)
33 WX_DEFINE_OBJARRAY(wxMethodArray
)
34 WX_DEFINE_OBJARRAY(wxClassArray
)
37 #define PROGRESS_RATE 1000 // each PROGRESS_RATE nodes processed print a dot
38 #define ESTIMATED_NUM_CLASSES 600 // used by both wxXmlInterface-derived classes to prealloc mem
41 // defined in ifacecheck.cpp
42 extern bool g_verbose
;
45 bool g_bLogEnabled
= true;
49 // ----------------------------------------------------------------------------
51 // ----------------------------------------------------------------------------
55 void wxType::SetTypeFromString(const wxString
& t
)
58 TODO: optimize the following code writing a single function
59 which works at char-level and does everything in a single pass
64 // [] is the same as * for gccxml
65 m_strType
.Replace("[]", "*");
66 m_strType
.Replace("long int", "long"); // in wx typically we never write "long int", just "long"
68 // make sure the * and & operator always use the same spacing rules
69 // (to make sure GetAsString() output is always consistent)
70 m_strType
.Replace("*", "* ");
71 m_strType
.Replace("&", "& ");
72 m_strType
.Replace(" *", "*");
73 m_strType
.Replace(" &", "&");
75 while (m_strType
.Contains(" "))
76 m_strType
.Replace(" ", " "); // do it once again
78 m_strType
.Replace(" ,", ",");
80 m_strType
= m_strType
.Strip(wxString::both
);
82 // now set the clean version
83 m_strTypeClean
= m_strType
;
84 m_strTypeClean
.Replace("const", "");
85 m_strTypeClean
.Replace("static", "");
86 m_strTypeClean
.Replace("*", "");
87 m_strTypeClean
.Replace("&", "");
88 m_strTypeClean
.Replace("[]", "");
89 m_strTypeClean
= m_strTypeClean
.Strip(wxString::both
);
91 // to avoid false errors types like wxStandardPaths and wxStandardPathsBase
92 // need to be considered as the same type
93 if (m_strTypeClean
.EndsWith("Base"))
94 m_strTypeClean
= m_strTypeClean
.Left(m_strTypeClean
.Len()-4);
97 // doxygen likes to put wxDateTime:: in front of all wxDateTime enums;
98 // fix this to avoid false positives
99 m_strTypeClean
.Replace("wxDateTime::", "");
100 m_strTypeClean
.Replace("wxStockGDI::", ""); // same story for some other classes
101 m_strTypeClean
.Replace("wxHelpEvent::", "");
104 bool wxType::IsOk() const
106 // NB: m_strType can contain the :: operator; think to e.g. the
107 // "reverse_iterator_impl<wxString::const_iterator>" type
108 // It can also contain commas, * and & operators etc
110 return !m_strTypeClean
.IsEmpty();
113 bool wxType::operator==(const wxType
& m
) const
115 // brain-dead comparison:
117 if (m_strTypeClean
== m
.m_strTypeClean
&&
118 IsConst() == m
.IsConst() &&
119 IsStatic() == m
.IsStatic() &&
120 IsPointer() == m
.IsPointer() &&
121 IsReference() == m
.IsReference())
126 LogMessage("Type '%s' does not match type '%s'", m_strType
, m
.m_strType
);
127 LogMessage(" => TypeClean %s / %s; IsConst %d / %d; IsStatic %d / %d; IsPointer %d / %d; IsReference %d / %d",
128 m_strTypeClean
, m
.m_strTypeClean
, IsConst(), m
.IsConst(),
129 IsStatic(), m
.IsStatic(), IsPointer(), m
.IsPointer(),
130 IsReference(), m
.IsReference());
137 // ----------------------------------------------------------------------------
139 // ----------------------------------------------------------------------------
141 void wxArgumentType::SetDefaultValue(const wxString
& defval
, const wxString
& defvalForCmp
)
143 m_strDefaultValue
= defval
.Strip(wxString::both
);
144 m_strDefaultValueForCmp
= defvalForCmp
.IsEmpty() ? m_strDefaultValue
: defvalForCmp
.Strip(wxString::both
);
146 // adjust aesthetic form of DefaultValue for the modify mode of ifacecheck:
147 // we may need to write it out in an interface header
148 if (m_strDefaultValue
== "0u")
149 m_strDefaultValue
= "0";
151 // in order to make valid&simple comparison on argument defaults,
152 // we reduce some of the multiple forms in which the same things may appear
154 if (m_strDefaultValueForCmp
== "0u")
155 m_strDefaultValueForCmp
= "0";
157 m_strDefaultValue
.Replace("0x000000001", "1");
158 m_strDefaultValueForCmp
.Replace("0x000000001", "1");
160 // fix for unicode strings:
161 m_strDefaultValue
.Replace("\\000\\000\\000", "");
162 m_strDefaultValueForCmp
.Replace("\\000\\000\\000", "");
164 if (m_strDefaultValueForCmp
.StartsWith("wxT(") &&
165 m_strDefaultValueForCmp
.EndsWith(")"))
167 // get rid of the wxT() part
168 unsigned int len
= m_strDefaultValueForCmp
.Len();
169 m_strDefaultValueForCmp
= m_strDefaultValueForCmp
.Mid(4,len
-5);
174 m_strDefaultValueForCmp.Replace("0", "NULL");
176 m_strDefaultValueForCmp.Replace("NULL", "0");
179 // doxygen likes to put wxDateTime:: in front of all wxDateTime enums;
180 // fix this to avoid false positives
181 m_strDefaultValueForCmp
.Replace("wxDateTime::", "");
182 m_strDefaultValueForCmp
.Replace("wxStockGDI::", ""); // same story for some other classes
183 m_strDefaultValueForCmp
.Replace("wxHelpEvent::", ""); // same story for some other classes
185 m_strDefaultValueForCmp
.Replace("wxGet_wxConvLocal()", "wxConvLocal");
187 m_strDefaultValueForCmp
.Replace("* GetColour(COLOUR_BLACK)", "*wxBLACK");
190 if (m_strDefaultValueForCmp
.Contains("wxGetTranslation"))
191 m_strDefaultValueForCmp
= "_(TOFIX)"; // TODO: wxGetTranslation gives problems to gccxml
194 bool wxArgumentType::operator==(const wxArgumentType
& m
) const
196 if ((const wxType
&)(*this) != (const wxType
&)m
)
200 // default values for style attributes of wxWindow-derived classes in gccxml appear as raw
201 // numbers; avoid false positives in this case!
202 if (m_strArgName
== m
.m_strArgName
&& m_strArgName
== "style" &&
203 (m_strDefaultValueForCmp
.IsNumber() || m
.m_strDefaultValueForCmp
.IsNumber()))
206 // fix for default values which were replaced by gcc-xml with their numeric values
207 // (at this point we know that m_strTypeClean == m.m_strTypeClean):
208 if (m_strTypeClean
== "long" || m_strTypeClean
== "int")
210 if ((m_strDefaultValueForCmp
.IsNumber() && m
.m_strDefaultValueForCmp
.StartsWith("wx")) ||
211 (m
.m_strDefaultValueForCmp
.IsNumber() && m_strDefaultValueForCmp
.StartsWith("wx")))
214 LogMessage("Supposing '%s' default value to be the same of '%s'...",
215 m_strDefaultValueForCmp
, m
.m_strDefaultValueForCmp
);
221 if (m_strDefaultValueForCmp
!= m
.m_strDefaultValueForCmp
)
223 // maybe the default values are numbers.
224 // in this case gccXML gives as default values things like '-0x0000001' instead of just '-1'.
225 // To handle these cases, we try to convert the default value strings to numbers:
226 long def1val
, def2val
;
227 if (m_strDefaultValueForCmp
.ToLong(&def1val
, 0 /* auto-detect */) &&
228 m
.m_strDefaultValueForCmp
.ToLong(&def2val
, 0 /* auto-detect */))
230 if (def1val
== def2val
)
231 return true; // the default values match
235 LogMessage("Argument type '%s = %s' has different default value from '%s = %s'",
236 m_strType
, m_strDefaultValueForCmp
, m
.m_strType
, m
.m_strDefaultValueForCmp
);
240 // we deliberately avoid checks on the argument name
246 // ----------------------------------------------------------------------------
248 // ----------------------------------------------------------------------------
250 bool wxMethod::IsOk() const
252 // NOTE: m_retType can be a wxEmptyType, and means that this method
253 // is a ctor or a dtor.
254 if (!m_retType
.IsOk() && m_retType
!=wxEmptyType
) {
255 LogError("'%s' method has invalid return type: %s", m_retType
.GetAsString());
259 if (m_strName
.IsEmpty())
262 // a function can't be both const and static or virtual and static!
263 if ((m_bConst
&& m_bStatic
) || ((m_bVirtual
|| m_bPureVirtual
) && m_bStatic
)) {
264 LogError("'%s' method can't be both const/static or virtual/static", m_strName
);
268 wxASSERT(!m_bPureVirtual
|| (m_bPureVirtual
&& m_bVirtual
));
270 for (unsigned int i
=0; i
<m_args
.GetCount(); i
++)
271 if (!m_args
[i
].IsOk()) {
272 LogError("'%s' method has invalid %d-th argument type: %s",
273 m_strName
, i
+1, m_args
[i
].GetAsString());
277 // NB: the default value of the arguments can contain pretty much everything
278 // (think to e.g. wxPoint(3+4/2,0) or *wxBLACK or someClass<type>)
279 // so we don't do any test on their contents
280 if (m_args
.GetCount()>0)
282 bool previousArgHasDefault
= m_args
[0].HasDefaultValue();
283 for (unsigned int i
=1; i
<m_args
.GetCount(); i
++)
285 if (previousArgHasDefault
&& !m_args
[i
].HasDefaultValue()) {
286 LogError("'%s' method has %d-th argument which has no default value "
287 "(while the previous one had one!)",
292 previousArgHasDefault
= m_args
[i
].HasDefaultValue();
299 bool wxMethod::MatchesExceptForAttributes(const wxMethod
& m
) const
301 if (GetReturnType() != m
.GetReturnType() ||
302 GetName() != m
.GetName())
305 LogMessage("The method '%s' does not match method '%s'; different names/rettype", GetName(), m
.GetName());
309 if (m_args
.GetCount()!=m
.m_args
.GetCount()) {
311 LogMessage("Method '%s' has %d arguments while '%s' has %d arguments",
312 m_strName
, m_args
.GetCount(), m_strName
, m
.m_args
.GetCount());
316 // compare argument types
317 for (unsigned int i
=0; i
<m_args
.GetCount(); i
++)
318 if (m_args
[i
] != m
.m_args
[i
])
324 bool wxMethod::operator==(const wxMethod
& m
) const
327 if (IsConst() != m
.IsConst() ||
328 IsStatic() != m
.IsStatic() ||
329 IsVirtual() != m
.IsVirtual() ||
330 IsPureVirtual() != m
.IsPureVirtual() ||
331 IsDeprecated() != m
.IsDeprecated() ||
332 GetAccessSpecifier() != m
.GetAccessSpecifier())
335 LogMessage("The method '%s' does not match method '%s'; different attributes", GetName(), m
.GetName());
340 // check everything else
341 return MatchesExceptForAttributes(m
);
344 wxString
wxMethod::GetAsString(bool bWithArgumentNames
, bool bCleanDefaultValues
,
345 bool bDeprecated
, bool bAccessSpec
) const
349 // NOTE: for return and argument types, never use wxType::GetAsCleanString
350 // since in that way we'd miss important decorators like &,*,const etc
352 if (m_retType
!=wxEmptyType
)
353 ret
+= m_retType
.GetAsString() + " ";
354 //else; this is a ctor or dtor
356 ret
+= m_strName
+ "(";
358 for (unsigned int i
=0; i
<m_args
.GetCount(); i
++)
360 ret
+= m_args
[i
].GetAsString();
362 const wxString
& name
= m_args
[i
].GetArgumentName();
363 if (bWithArgumentNames
&& !name
.IsEmpty())
366 const wxString
& def
= bCleanDefaultValues
?
367 m_args
[i
].GetDefaultCleanValue() : m_args
[i
].GetDefaultValue();
374 if (m_args
.GetCount()>0)
375 ret
= ret
.Left(ret
.Len()-2);
382 ret
= "static " + ret
;
383 if (m_bVirtual
|| m_bPureVirtual
)
384 ret
= "virtual " + ret
;
387 if (m_bDeprecated
&& bDeprecated
)
388 ret
+= " [deprecated]";
397 case wxMAS_PROTECTED
:
398 ret
+= " [protected]";
409 void wxMethod::Dump(wxTextOutputStream
& stream
) const
411 stream
<< "[" + m_retType
.GetAsString() + "]";
412 stream
<< "[" + m_strName
+ "]";
414 for (unsigned int i
=0; i
<m_args
.GetCount(); i
++)
415 stream
<< "[" + m_args
[i
].GetAsString() + " " + m_args
[i
].GetArgumentName() +
416 "=" + m_args
[i
].GetDefaultValue() + "]";
423 stream
<< " VIRTUAL";
425 stream
<< " PURE-VIRTUAL";
427 stream
<< " DEPRECATED";
432 // ----------------------------------------------------------------------------
434 // ----------------------------------------------------------------------------
436 wxString
wxClass::GetNameWithoutTemplate() const
438 // NB: I'm not sure this is the right terminology for this function!
440 if (m_strName
.Contains("<"))
441 return m_strName
.Left(m_strName
.Find("<"));
445 bool wxClass::IsValidCtorForThisClass(const wxMethod
& m
) const
447 // remember that e.g. the ctor for wxWritableCharTypeBuffer<wchar_t> is
448 // named wxWritableCharTypeBuffer, without the <...> part!
450 if (m
.IsCtor() && m
.GetName() == GetNameWithoutTemplate())
456 bool wxClass::IsValidDtorForThisClass(const wxMethod
& m
) const
458 if (m
.IsDtor() && m
.GetName() == "~" + GetNameWithoutTemplate())
464 void wxClass::Dump(wxTextOutputStream
& out
) const
466 out
<< m_strName
+ "\n";
468 for (unsigned int i
=0; i
<m_methods
.GetCount(); i
++) {
470 // dump all our methods
472 m_methods
[i
].Dump(out
);
479 bool wxClass::CheckConsistency() const
481 for (unsigned int i
=0; i
<m_methods
.GetCount(); i
++)
482 for (unsigned int j
=0; j
<m_methods
.GetCount(); j
++)
483 if (i
!=j
&& m_methods
[i
] == m_methods
[j
])
485 LogError("class %s has two methods with the same prototype: '%s'",
486 m_strName
, m_methods
[i
].GetAsString());
490 //((wxClass*)this)->m_methods.RemoveAt(j);
497 const wxMethod
* wxClass::FindMethod(const wxMethod
& m
) const
499 for (unsigned int i
=0; i
<m_methods
.GetCount(); i
++)
500 if (m_methods
[i
] == m
)
501 return &m_methods
[i
];
505 const wxMethod
* wxClass::RecursiveUpwardFindMethod(const wxMethod
& m
,
506 const wxXmlInterface
* allclasses
) const
508 // first, search into *this
509 const wxMethod
* ret
= FindMethod(m
);
513 // then, search into its parents
514 for (unsigned int i
=0; i
<m_parents
.GetCount(); i
++)
516 // ignore non-wx-classes parents
517 // AD-HOC FIX: discard wxScrolledT_Helper parent as it always gives errors
518 if (m_parents
[i
].StartsWith("wx") && m_parents
[i
] != "wxScrolledT_Helper")
520 const wxClass
*parent
= allclasses
->FindClass(m_parents
[i
]);
522 LogError("Could not find parent '%s' of class '%s'...",
523 m_parents
[i
], GetName());
527 const wxMethod
*parentMethod
= parent
->RecursiveUpwardFindMethod(m
, allclasses
);
533 // could not find anything even in parent classes...
537 wxMethodPtrArray
wxClass::FindMethodsNamed(const wxString
& name
) const
539 wxMethodPtrArray ret
;
541 for (unsigned int i
=0; i
<m_methods
.GetCount(); i
++)
542 if (m_methods
[i
].GetName() == name
)
543 ret
.Add(&m_methods
[i
]);
549 wxMethodPtrArray
wxClass::RecursiveUpwardFindMethodsNamed(const wxString
& name
,
550 const wxXmlInterface
* allclasses
) const
552 // first, search into *this
553 wxMethodPtrArray ret
= FindMethodsNamed(name
);
554 if (ret
.GetCount()>0)
555 return ret
; // stop here, don't look upward in the parents
557 // then, search into parents of this class
558 for (unsigned int i
=0; i
<m_parents
.GetCount(); i
++)
560 // AD-HOC FIX: discard wxScrolledT_Helper parent as it always gives errors
561 if (m_parents
[i
].StartsWith("wx") && m_parents
[i
] != "wxScrolledT_Helper")
563 const wxClass
*parent
= allclasses
->FindClass(m_parents
[i
]);
565 LogError("Could not find parent '%s' of class '%s'...",
566 m_parents
[i
], GetName());
570 wxMethodPtrArray temp
= parent
->RecursiveUpwardFindMethodsNamed(name
, allclasses
);
571 WX_APPEND_ARRAY(ret
, temp
);
580 // ----------------------------------------------------------------------------
582 // ----------------------------------------------------------------------------
584 WX_DEFINE_SORTED_ARRAY(wxClass
*, wxSortedClassArray
);
586 int CompareWxClassObjects(wxClass
*item1
, wxClass
*item2
)
588 // sort alphabetically
589 return item1
->GetName().Cmp(item2
->GetName());
592 void wxXmlInterface::Dump(const wxString
& filename
)
594 wxFFileOutputStream
apioutput( filename
);
595 wxTextOutputStream
apiout( apioutput
);
597 // dump the classes in alphabetical order
598 wxSortedClassArray
sorted(CompareWxClassObjects
);
599 sorted
.Alloc(m_classes
.GetCount());
600 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
601 sorted
.Add(&m_classes
[i
]);
603 // now they have been sorted
604 for (unsigned int i
=0; i
<sorted
.GetCount(); i
++)
605 sorted
[i
]->Dump(apiout
);
608 bool wxXmlInterface::CheckParseResults() const
610 // this check can be quite slow, so do it only for debug releases:
612 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
613 if (!m_classes
[i
].CheckConsistency())
620 wxClassPtrArray
wxXmlInterface::FindClassesDefinedIn(const wxString
& headerfile
) const
624 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
625 if (m_classes
[i
].GetHeader() == headerfile
)
626 ret
.Add(&m_classes
[i
]);
632 // ----------------------------------------------------------------------------
633 // wxXmlGccInterface helper declarations
634 // ----------------------------------------------------------------------------
636 // or-able flags for a toResolveTypeItem->attrib:
637 #define ATTRIB_CONST 1
638 #define ATTRIB_REFERENCE 2
639 #define ATTRIB_POINTER 4
640 #define ATTRIB_ARRAY 8
642 // it may sound strange but gccxml, in order to produce shorter ID names
643 // uses (after the underscore) characters in range 0-9 and a-z in the ID names;
644 // in order to be able to translate such strings into numbers using strtoul()
645 // we use as base 10 (possible digits) + 25 (possible characters) = 35
646 #define GCCXML_BASE 35
648 class toResolveTypeItem
651 toResolveTypeItem() { attribs
=0; }
652 toResolveTypeItem(unsigned int refID
, unsigned int attribint
)
653 : ref(refID
), attribs(attribint
) {}
655 unsigned long ref
, // the referenced type's ID
656 attribs
; // the attributes of this reference
661 // for wxToResolveTypeHashMap, keys == gccXML IDs and values == toResolveTypeItem
662 WX_DECLARE_HASH_MAP( unsigned long, toResolveTypeItem
,
663 wxIntegerHash
, wxIntegerEqual
,
664 wxToResolveTypeHashMap
);
666 // for wxClassMemberIdHashMap, keys == gccXML IDs and values == wxClass which owns that member ID
667 WX_DECLARE_HASH_MAP( unsigned long, wxClass
*,
668 wxIntegerHash
, wxIntegerEqual
,
669 wxClassMemberIdHashMap
);
673 typedef std::map
<unsigned long, toResolveTypeItem
> wxToResolveTypeHashMap
;
677 // utility to parse gccXML ID values;
678 // this function is equivalent to wxString(str).Mid(1).ToULong(&id, GCCXML_BASE)
679 // but is a little bit faster
680 bool getID(unsigned long *id
, const wxString
& str
)
682 const wxStringCharType
* const start
= str
.wx_str()+1;
683 wxStringCharType
*end
;
684 #if wxUSE_UNICODE_WCHAR
685 unsigned long val
= wcstoul(start
, &end
, GCCXML_BASE
);
687 unsigned long val
= strtoul(start
, &end
, GCCXML_BASE
);
690 // return true only if scan was stopped by the terminating NUL and
691 // if the string was not empty to start with and no under/overflow
693 if ( *end
!= '\0' || end
== start
|| errno
== ERANGE
|| errno
== EINVAL
)
700 // utility specialized to parse efficiently the gccXML list of IDs which occur
701 // in nodes like <Class> ones... i.e. numeric values separed by " _" token
702 bool getMemberIDs(wxClassMemberIdHashMap
* map
, wxClass
* p
, const wxString
& str
)
704 const wxStringCharType
* const start
= str
.wx_str();
705 #if wxUSE_UNICODE_WCHAR
706 size_t len
= wcslen(start
);
708 size_t len
= strlen(start
);
711 if (len
== 0 || start
[0] != '_')
714 const wxStringCharType
*curpos
= start
,
716 wxStringCharType
*nexttoken
;
720 // curpos always points to the underscore of the next token to parse:
721 #if wxUSE_UNICODE_WCHAR
722 unsigned long id
= wcstoul(curpos
+1, &nexttoken
, GCCXML_BASE
);
724 unsigned long id
= strtoul(curpos
+1, &nexttoken
, GCCXML_BASE
);
726 if ( *nexttoken
!= ' ' || errno
== ERANGE
|| errno
== EINVAL
)
729 // advance current position
730 curpos
= nexttoken
+ 1;
732 // add this ID to the hashmap
733 wxClassMemberIdHashMap::value_type
v(id
, p
);
741 // ----------------------------------------------------------------------------
743 // ----------------------------------------------------------------------------
745 bool wxXmlGccInterface::Parse(const wxString
& filename
)
751 LogMessage("Parsing %s...", filename
);
753 if (!doc
.Load(filename
)) {
754 LogError("can't load %s", filename
);
758 // start processing the XML file
759 if (doc
.GetRoot()->GetName() != "GCC_XML") {
760 LogError("invalid root node for %s", filename
);
764 wxString version
= doc
.GetRoot()->GetAttribute("cvs_revision");
767 #define MIN_REVISION 120
769 if (!version
.StartsWith("1."))
773 unsigned long rev
= 0;
774 if (!version
.Mid(2).ToULong(&rev
))
777 if (rev
< MIN_REVISION
)
783 LogError("The version of GCC-XML used for the creation of %s is too old; "
784 "the cvs_revision attribute of the root node reports '%s', "
785 "minimal required is 1.%d.", filename
, version
, MIN_REVISION
);
789 wxToResolveTypeHashMap toResolveTypes
;
790 wxClassMemberIdHashMap members
;
791 wxTypeIdHashMap types
;
792 wxTypeIdHashMap files
;
793 wxTypeIdHashMap typedefs
;
795 // prealloc quite a lot of memory!
796 m_classes
.Alloc(ESTIMATED_NUM_CLASSES
);
798 // build a list of wx classes and in general of all existent types
799 child
= doc
.GetRoot()->GetChildren();
802 const wxString
& n
= child
->GetName();
804 unsigned long id
= 0;
805 if (!getID(&id
, child
->GetAttribute("id")) || (id
== 0 && n
!= "File")) {
807 // NOTE: <File> nodes can have an id == "f0"...
809 LogError("Invalid id for node %s: %s", n
, child
->GetAttribute("id"));
815 wxString cname
= child
->GetAttribute("name");
816 if (cname
.IsEmpty()) {
817 LogError("Invalid empty name for '%s' node", n
);
821 // only register wx classes (do remember also the IDs of their members)
822 if (cname
.StartsWith("wx"))
824 // NB: "file" attribute contains an ID value that we'll resolve later
825 m_classes
.Add(wxClass(cname
, child
->GetAttribute("file")));
827 // the just-inserted class:
828 wxClass
*newClass
= &m_classes
.Last();
830 // now get a list of the base classes:
831 wxXmlNode
*baseNode
= child
->GetChildren();
834 // for now we store as "parents" only the parent IDs...
835 // later we will resolve them into full class names
836 if (baseNode
->GetName() == "Base")
837 newClass
->AddParent(baseNode
->GetAttribute("type"));
839 baseNode
= baseNode
->GetNext();
842 const wxString
& ids
= child
->GetAttribute("members");
845 if (child
->GetAttribute("incomplete") != "1") {
846 LogError("Invalid member IDs for '%s' class node: %s",
847 cname
, child
->GetAttribute("id"));
850 //else: don't warn the user; it looks like "incomplete" classes
851 // never have any member...
855 // decode the non-empty list of IDs:
856 if (!getMemberIDs(&members
, newClass
, ids
)) {
857 LogError("Invalid member IDs for '%s' class node: %s",
858 cname
, child
->GetAttribute("id"));
864 // register this class also as possible return/argument type:
867 else if (n
== "Typedef")
869 unsigned long typeId
= 0;
870 if (!getID(&typeId
, child
->GetAttribute("type"))) {
871 LogError("Invalid type for node %s: %s", n
, child
->GetAttribute("type"));
875 // this typedef node tell us that every type referenced with the
876 // "typeId" ID should be called with another name:
877 wxString name
= child
->GetAttribute("name");
879 // save this typedef in a separate hashmap...
880 typedefs
[typeId
] = name
;
884 else if (n
== "PointerType" || n
== "ReferenceType" ||
885 n
== "CvQualifiedType" || n
== "ArrayType")
887 unsigned long type
= 0;
888 if (!getID(&type
, child
->GetAttribute("type")) || type
== 0) {
889 LogError("Invalid type for node %s: %s", n
, child
->GetAttribute("type"));
893 unsigned long attr
= 0;
894 if (n
== "PointerType")
895 attr
= ATTRIB_POINTER
;
896 else if (n
== "ReferenceType")
897 attr
= ATTRIB_REFERENCE
;
898 else if (n
== "CvQualifiedType" && child
->GetAttribute("const") == "1")
900 else if (n
== "ArrayType")
903 // these nodes make reference to other types... we'll resolve them later
904 toResolveTypes
[id
] = toResolveTypeItem(type
, attr
);
906 else if (n
== "FunctionType" || n
== "MethodType")
909 TODO: parsing FunctionType and MethodType nodes is not as easy
910 as for other "simple" types.
914 wxXmlNode
*arg
= child
->GetChildren();
917 if (arg
->GetName() == "Argument")
918 argstr
+= arg
->GetAttribute("type") + ", ";
919 arg
= arg
->GetNext();
922 if (argstr
.Len() > 0)
923 argstr
= argstr
.Left(argstr
.Len()-2); // remove final comma
925 // these nodes make reference to other types... we'll resolve them later
926 //toResolveTypes[id] = toResolveTypeItem(ret, 0);
927 //types[id] = child->GetAttribute("returns") + "(" + argstr + ")";
929 types
[id
] = "TOFIX"; // typically this type will be "fixed" thanks
930 // to a typedef later...
932 else if (n
== "File")
934 if (!child
->GetAttribute("id").StartsWith("f")) {
935 LogError("Unexpected file ID: %s", child
->GetAttribute("id"));
939 // just ignore this node... all file IDs/names were already parsed
940 files
[id
] = child
->GetAttribute("name");
944 // we register everything else as a possible return/argument type:
945 const wxString
& name
= child
->GetAttribute("name");
950 //typeNames.Add(name);
955 // this may happen with unnamed structs/union, special ctors,
956 // or other exotic things which we are not interested to, since
957 // they're never used as return/argument types by wxWidgets methods
960 LogWarning("Type node '%s' with ID '%s' does not have name attribute",
961 n
, child
->GetAttribute("id"));
967 child
= child
->GetNext();
969 // give feedback to the user about the progress...
970 if ((++nodes%PROGRESS_RATE
)==0) ShowProgress();
973 // some nodes with IDs referenced by methods as return/argument types, do reference
974 // in turn other nodes (see PointerType, ReferenceType and CvQualifierType above);
975 // thus we need to resolve their name iteratively:
976 while (toResolveTypes
.size()>0)
979 LogMessage("%d types were collected; %d types need yet to be resolved...",
980 types
.size(), toResolveTypes
.size());
982 for (wxToResolveTypeHashMap::iterator i
= toResolveTypes
.begin();
983 i
!= toResolveTypes
.end();)
985 unsigned long id
= i
->first
;
986 unsigned long referenced
= i
->second
.ref
;
988 wxTypeIdHashMap::iterator primary
= types
.find(referenced
);
989 if (primary
!= types
.end())
991 // this to-resolve-type references a "primary" type
993 wxString newtype
= primary
->second
;
994 int attribs
= i
->second
.attribs
;
996 // attribs may contain a combination of ATTRIB_* flags:
997 if (attribs
& ATTRIB_CONST
)
998 newtype
= "const " + newtype
;
999 if (attribs
& ATTRIB_REFERENCE
)
1000 newtype
= newtype
+ "&";
1001 if (attribs
& ATTRIB_POINTER
)
1002 newtype
= newtype
+ "*";
1003 if (attribs
& ATTRIB_ARRAY
)
1004 newtype
= newtype
+ "[]";
1006 // add the resolved type to the list of "primary" types
1007 if (newtype
.Contains("TOFIX") && typedefs
[id
] != "")
1008 types
[id
] = typedefs
[id
]; // better use a typedef for this type!
1010 types
[id
] = newtype
;
1012 // this one has been resolved; erase it through its iterator!
1013 toResolveTypes
.erase(i
);
1015 // now iterator i is invalid; assign it again to the beginning
1016 i
= toResolveTypes
.begin();
1020 // then search in the referenced types themselves:
1021 wxToResolveTypeHashMap::iterator idx2
= toResolveTypes
.find(referenced
);
1022 if (idx2
!= toResolveTypes
.end())
1024 // merge this to-resolve-type with the idx2->second type
1025 i
->second
.ref
= idx2
->second
.ref
;
1026 i
->second
.attribs
|= idx2
->second
.attribs
;
1028 // this type will eventually be solved in the next while() iteration
1033 LogError("Cannot solve '%d' reference type!", referenced
);
1040 // resolve header names
1041 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
1043 unsigned long fileID
= 0;
1044 if (!getID(&fileID
, m_classes
[i
].GetHeader()) || fileID
== 0) {
1045 LogError("invalid header id: %s", m_classes
[i
].GetHeader());
1050 wxTypeIdHashMap::const_iterator idx
= files
.find(fileID
);
1051 if (idx
== files
.end())
1053 // this is an error!
1054 LogError("couldn't find file ID '%s'", m_classes
[i
].GetHeader());
1057 m_classes
[i
].SetHeader(idx
->second
);
1060 // resolve parent names
1061 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
1063 for (unsigned int k
=0; k
<m_classes
[i
].GetParentCount(); k
++)
1067 if (!getID(&id
, m_classes
[i
].GetParent(k
))) {
1068 LogError("invalid parent class ID for '%s'", m_classes
[i
].GetName());
1072 wxTypeIdHashMap::const_iterator idx
= types
.find(id
);
1073 if (idx
== types
.end())
1075 // this is an error!
1076 LogError("couldn't find parent class ID '%d'", id
);
1079 // replace k-th parent with its true name:
1080 m_classes
[i
].SetParent(k
, idx
->second
);
1084 // build the list of the wx methods
1085 child
= doc
.GetRoot()->GetChildren();
1088 wxString n
= child
->GetName(), acc
= child
->GetAttribute("access");
1090 // only register public&protected methods
1091 if ((acc
== "public" || acc
== "protected") &&
1092 (n
== "Method" || n
== "Constructor" || n
== "Destructor" || n
== "OperatorMethod"))
1094 unsigned long id
= 0;
1095 if (!getID(&id
, child
->GetAttribute("id"))) {
1096 LogError("invalid ID for node '%s' with ID '%s'", n
, child
->GetAttribute("id"));
1100 wxClassMemberIdHashMap::const_iterator it
= members
.find(id
);
1101 if (it
!= members
.end())
1103 wxClass
*p
= it
->second
;
1105 // this <Method> node is a method of the i-th class!
1107 if (!ParseMethod(child
, types
, newfunc
)) {
1108 LogError("The method '%s' could not be added to class '%s'",
1109 child
->GetAttribute("demangled"), p
->GetName());
1113 // do some additional check that we can do only here:
1115 if (newfunc
.IsCtor() && !p
->IsValidCtorForThisClass(newfunc
)) {
1116 LogError("The method '%s' does not seem to be a ctor for '%s'",
1117 newfunc
.GetName(), p
->GetName());
1120 if (newfunc
.IsDtor() && !p
->IsValidDtorForThisClass(newfunc
)) {
1121 LogError("The method '%s' does not seem to be a dtor for '%s'",
1122 newfunc
.GetName(), p
->GetName());
1126 p
->AddMethod(newfunc
);
1130 child
= child
->GetNext();
1132 // give feedback to the user about the progress...
1133 if ((++nodes%PROGRESS_RATE
)==0) ShowProgress();
1136 if (!CheckParseResults())
1142 bool wxXmlGccInterface::ParseMethod(const wxXmlNode
*p
,
1143 const wxTypeIdHashMap
& types
,
1146 // get the real name
1147 wxString name
= p
->GetAttribute("name").Strip(wxString::both
);
1148 if (p
->GetName() == "Destructor")
1150 else if (p
->GetName() == "OperatorMethod")
1151 name
= "operator" + name
;
1153 // resolve return type
1155 unsigned long retid
= 0;
1156 if (!getID(&retid
, p
->GetAttribute("returns")) || retid
== 0)
1158 if (p
->GetName() != "Destructor" && p
->GetName() != "Constructor") {
1159 LogError("Empty return ID for method '%s', with ID '%s'",
1160 name
, p
->GetAttribute("id"));
1166 wxTypeIdHashMap::const_iterator retidx
= types
.find(retid
);
1167 if (retidx
== types
.end()) {
1168 LogError("Could not find return type ID '%s'", retid
);
1172 ret
= wxType(retidx
->second
);
1174 LogError("Invalid return type '%s' for method '%s', with ID '%s'",
1175 retidx
->second
, name
, p
->GetAttribute("id"));
1180 // resolve argument types
1181 wxArgumentTypeArray argtypes
;
1182 wxXmlNode
*arg
= p
->GetChildren();
1185 if (arg
->GetName() == "Argument")
1187 unsigned long id
= 0;
1188 if (!getID(&id
, arg
->GetAttribute("type")) || id
== 0) {
1189 LogError("Invalid argument type ID '%s' for method '%s' with ID %s",
1190 arg
->GetAttribute("type"), name
, p
->GetAttribute("id"));
1194 wxTypeIdHashMap::const_iterator idx
= types
.find(id
);
1195 if (idx
== types
.end()) {
1196 LogError("Could not find argument type ID '%s'", id
);
1200 argtypes
.Add(wxArgumentType(idx
->second
,
1201 arg
->GetAttribute("default"),
1202 arg
->GetAttribute("name")));
1205 arg
= arg
->GetNext();
1208 m
.SetReturnType(ret
);
1210 m
.SetArgumentTypes(argtypes
);
1211 m
.SetConst(p
->GetAttribute("const") == "1");
1212 m
.SetStatic(p
->GetAttribute("static") == "1");
1214 // NOTE: gccxml is smart enough to mark as virtual those functions
1215 // which are declared virtual in base classes but don't have
1216 // the "virtual" keyword explicitely indicated in the derived
1217 // classes... so we don't need any further logic for virtuals
1219 m
.SetVirtual(p
->GetAttribute("virtual") == "1");
1220 m
.SetPureVirtual(p
->GetAttribute("pure_virtual") == "1");
1221 m
.SetDeprecated(p
->GetAttribute("attributes") == "deprecated");
1223 // decode access specifier
1224 if (p
->GetAttribute("access") == "public")
1225 m
.SetAccessSpecifier(wxMAS_PUBLIC
);
1226 else if (p
->GetAttribute("access") == "protected")
1227 m
.SetAccessSpecifier(wxMAS_PROTECTED
);
1228 else if (p
->GetAttribute("access") == "private")
1229 m
.SetAccessSpecifier(wxMAS_PRIVATE
);
1232 LogError("The prototype '%s' is not valid!", m
.GetAsString());
1241 // ----------------------------------------------------------------------------
1242 // wxXmlDoxygenInterface global helpers
1243 // ----------------------------------------------------------------------------
1245 static wxString
GetTextFromChildren(const wxXmlNode
*n
)
1249 // consider the tree
1251 // <a><b>this</b> is a <b>string</b></a>
1260 // unlike wxXmlNode::GetNodeContent() which would return " is a "
1261 // this function returns "this is a string"
1263 wxXmlNode
*ref
= n
->GetChildren();
1265 if (ref
->GetType() == wxXML_ELEMENT_NODE
)
1266 text
+= ref
->GetNodeContent();
1267 else if (ref
->GetType() == wxXML_TEXT_NODE
)
1268 text
+= ref
->GetContent();
1270 LogWarning("Unexpected node type while getting text from '%s' node", n
->GetName());
1272 ref
= ref
->GetNext();
1278 static bool HasTextNodeContaining(const wxXmlNode
*parent
, const wxString
& name
)
1283 wxXmlNode
*p
= parent
->GetChildren();
1286 switch (p
->GetType())
1288 case wxXML_TEXT_NODE
:
1289 if (p
->GetContent() == name
)
1293 case wxXML_ELEMENT_NODE
:
1294 // recurse into this node...
1295 if (HasTextNodeContaining(p
, name
))
1310 static const wxXmlNode
* FindNodeNamed(const wxXmlNode
* parent
, const wxString
& name
)
1315 const wxXmlNode
*p
= parent
->GetChildren();
1318 if (p
->GetName() == name
)
1321 // search recursively in the children of this node
1322 const wxXmlNode
*ret
= FindNodeNamed(p
, name
);
1332 int GetAvailabilityFor(const wxXmlNode
*node
)
1334 // identify <onlyfor> custom XML tags
1335 const wxXmlNode
* onlyfor
= FindNodeNamed(node
, "onlyfor");
1337 return wxPORT_UNKNOWN
;
1339 wxArrayString ports
= wxSplit(onlyfor
->GetNodeContent(), ',');
1340 int nAvail
= wxPORT_UNKNOWN
;
1341 for (unsigned int i
=0; i
< ports
.GetCount(); i
++)
1343 if (!ports
[i
].StartsWith("wx")) {
1344 LogError("unexpected port ID '%s'", ports
[i
]);
1348 nAvail
|= wxPlatformInfo::GetPortId(ports
[i
].Mid(2));
1355 // ----------------------------------------------------------------------------
1356 // wxXmlDoxygenInterface
1357 // ----------------------------------------------------------------------------
1359 bool wxXmlDoxygenInterface::Parse(const wxString
& filename
)
1361 wxXmlDocument index
;
1362 wxXmlNode
*compound
;
1364 LogMessage("Parsing %s...", filename
);
1366 if (!index
.Load(filename
)) {
1367 LogError("can't load %s", filename
);
1371 // start processing the index:
1372 if (index
.GetRoot()->GetName() != "doxygenindex") {
1373 LogError("invalid root node for %s", filename
);
1378 NB: we may need in future to do a version-check here if the
1379 format of the XML generated by doxygen changes.
1380 For now (doxygen version 1.5.5), this check is not required
1381 since AFAIK the XML format never changed since it was introduced.
1384 m_classes
.Alloc(ESTIMATED_NUM_CLASSES
);
1386 // process files referenced by this index file
1387 compound
= index
.GetRoot()->GetChildren();
1390 if (compound
->GetName() == "compound" &&
1391 compound
->GetAttribute("kind") == "class")
1393 wxString refid
= compound
->GetAttribute("refid");
1395 wxFileName
fn(filename
);
1396 if (!ParseCompoundDefinition(fn
.GetPath(wxPATH_GET_SEPARATOR
) + refid
+ ".xml"))
1400 compound
= compound
->GetNext();
1404 if (!CheckParseResults())
1410 bool wxXmlDoxygenInterface::ParseCompoundDefinition(const wxString
& filename
)
1412 wxClassMemberIdHashMap parents
;
1418 LogMessage("Parsing %s...", filename
);
1420 if (!doc
.Load(filename
)) {
1421 LogError("can't load %s", filename
);
1425 // start processing this compound definition XML
1426 if (doc
.GetRoot()->GetName() != "doxygen") {
1427 LogError("invalid root node for %s", filename
);
1431 // build a list of wx classes
1432 child
= doc
.GetRoot()->GetChildren();
1435 if (child
->GetName() == "compounddef" &&
1436 child
->GetAttribute("kind") == "class")
1440 wxString absoluteFile
, header
;
1442 wxXmlNode
*subchild
= child
->GetChildren();
1445 wxString kind
= subchild
->GetAttribute("kind");
1447 // parse only public&protected functions:
1448 if (subchild
->GetName() == "sectiondef" &&
1449 (kind
== "public-func" || kind
== "protected-func"))
1452 wxXmlNode
*membernode
= subchild
->GetChildren();
1455 if (membernode
->GetName() == "memberdef" &&
1456 membernode
->GetAttribute("kind") == "function")
1460 if (!ParseMethod(membernode
, m
, header
)) {
1461 LogError("The method '%s' could not be added to class '%s'",
1462 m
.GetName(), klass
.GetName());
1466 if (kind
== "public-func")
1467 m
.SetAccessSpecifier(wxMAS_PUBLIC
);
1468 else if (kind
== "protected-func")
1469 m
.SetAccessSpecifier(wxMAS_PROTECTED
);
1470 else if (kind
== "private-func")
1471 m
.SetAccessSpecifier(wxMAS_PRIVATE
);
1473 if (absoluteFile
.IsEmpty())
1474 absoluteFile
= header
;
1475 else if (header
!= absoluteFile
)
1477 LogError("The method '%s' is documented in a different "
1478 "file from others (which belong to '%s') ?",
1479 header
, absoluteFile
);
1486 membernode
= membernode
->GetNext();
1489 // all methods of this class were taken from the header "absoluteFile":
1490 klass
.SetHeader(absoluteFile
);
1492 else if (subchild
->GetName() == "compoundname")
1494 klass
.SetName(subchild
->GetNodeContent());
1496 /*else if (subchild->GetName() == "includes")
1498 // NOTE: we'll get the header from the <location> tags
1499 // scattered inside <memberdef> tags instead of
1500 // this <includes> tag since it does not contain
1501 // the absolute path of the header
1503 klass.SetHeader(subchild->GetNodeContent());
1505 else if (subchild
->GetName() == "detaileddescription")
1507 // identify <onlyfor> custom XML tags
1508 klass
.SetAvailability(GetAvailabilityFor(subchild
));
1510 else if (subchild
->GetName() == "basecompoundref")
1512 // add the name of this parent to the list of klass' parents
1513 klass
.AddParent(subchild
->GetNodeContent());
1516 subchild
= subchild
->GetNext();
1521 m_classes
.Add(klass
);
1523 LogWarning("discarding class '%s' with %d methods...",
1524 klass
.GetName(), klass
.GetMethodCount());
1527 child
= child
->GetNext();
1529 // give feedback to the user about the progress...
1530 if ((++nodes%PROGRESS_RATE
)==0) ShowProgress();
1536 bool wxXmlDoxygenInterface::ParseMethod(const wxXmlNode
* p
, wxMethod
& m
, wxString
& header
)
1538 wxArgumentTypeArray args
;
1541 wxXmlNode
*child
= p
->GetChildren();
1544 if (child
->GetName() == "name")
1545 m
.SetName(child
->GetNodeContent());
1546 else if (child
->GetName() == "type")
1547 m
.SetReturnType(wxType(GetTextFromChildren(child
)));
1548 else if (child
->GetName() == "param")
1550 wxString typestr
, namestr
, defstr
, arrstr
;
1551 wxXmlNode
*n
= child
->GetChildren();
1554 if (n
->GetName() == "type")
1555 // if the <type> node has children, they should be all TEXT and <ref> nodes
1556 // and we need to take the text they contain, in the order they appear
1557 typestr
= GetTextFromChildren(n
);
1558 else if (n
->GetName() == "declname")
1559 namestr
= GetTextFromChildren(n
);
1560 else if (n
->GetName() == "defval")
1561 defstr
= GetTextFromChildren(n
).Strip(wxString::both
);
1562 else if (n
->GetName() == "array")
1563 arrstr
= GetTextFromChildren(n
);
1568 if (typestr
.IsEmpty()) {
1569 LogError("cannot find type node for a param in method '%s'", m
.GetName());
1573 wxArgumentType
newarg(typestr
+ arrstr
, defstr
, namestr
);
1575 // can we use preprocessor output to transform the default value
1576 // into the same form which gets processed by wxXmlGccInterface?
1577 wxStringHashMap::const_iterator it
= m_preproc
.find(defstr
);
1578 if (it
!= m_preproc
.end())
1579 newarg
.SetDefaultValue(defstr
, it
->second
);
1583 else if (child
->GetName() == "location")
1586 if (child
->GetAttribute("line").ToLong(&line
))
1587 m
.SetLocation((int)line
);
1588 header
= child
->GetAttribute("file");
1590 else if (child
->GetName() == "detaileddescription")
1592 // when a method has a @deprecated tag inside its description,
1593 // Doxygen outputs somewhere nested inside <detaileddescription>
1594 // a <xreftitle>Deprecated</xreftitle> tag.
1595 m
.SetDeprecated(HasTextNodeContaining(child
, "Deprecated"));
1597 // identify <onlyfor> custom XML tags
1598 m
.SetAvailability(GetAvailabilityFor(child
));
1601 child
= child
->GetNext();
1604 m
.SetArgumentTypes(args
);
1605 m
.SetConst(p
->GetAttribute("const")=="yes");
1606 m
.SetStatic(p
->GetAttribute("static")=="yes");
1608 // NOTE: Doxygen is smart enough to mark as virtual those functions
1609 // which are declared virtual in base classes but don't have
1610 // the "virtual" keyword explicitely indicated in the derived
1611 // classes... so we don't need any further logic for virtuals
1613 m
.SetVirtual(p
->GetAttribute("virt")=="virtual");
1614 m
.SetPureVirtual(p
->GetAttribute("virt")=="pure-virtual");
1617 LogError("The prototype '%s' is not valid!", m
.GetAsString());