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
62 // clean the type string
63 // ---------------------
67 // [] is the same as * for gccxml
68 m_strType
.Replace("[]", "*");
69 m_strType
.Replace("long int", "long"); // in wx typically we never write "long int", just "long"
70 m_strType
.Replace("long unsigned int", "unsigned long");
72 // make sure the * and & operator always use the same spacing rules
73 // (to make sure GetAsString() output is always consistent)
74 m_strType
.Replace("*", "* ");
75 m_strType
.Replace("&", "& ");
76 m_strType
.Replace(" *", "*");
77 m_strType
.Replace(" &", "&");
79 while (m_strType
.Contains(" "))
80 m_strType
.Replace(" ", " "); // do it once again
82 m_strType
.Replace(" ,", ",");
84 m_strType
= m_strType
.Strip(wxString::both
);
88 // clean the type string (this time for the comparison)
89 // ----------------------------------------------------
91 m_strTypeClean
= m_strType
; // begin with the already-cleaned string
92 m_strTypeClean
.Replace("const", "");
93 m_strTypeClean
.Replace("static", "");
94 m_strTypeClean
.Replace("*", "");
95 m_strTypeClean
.Replace("&", "");
96 m_strTypeClean
.Replace("[]", "");
97 m_strTypeClean
= m_strTypeClean
.Strip(wxString::both
);
99 // to avoid false errors types like wxStandardPaths and wxStandardPathsBase
100 // need to be considered as the same type
101 if (m_strTypeClean
.EndsWith("Base"))
102 m_strTypeClean
= m_strTypeClean
.Left(m_strTypeClean
.Len()-4);
105 // doxygen likes to put wxDateTime:: in front of all wxDateTime enums;
106 // fix this to avoid false positives
107 m_strTypeClean
.Replace("wxDateTime::", "");
108 m_strTypeClean
.Replace("wxStockGDI::", ""); // same story for some other classes
109 m_strTypeClean
.Replace("wxHelpEvent::", "");
112 bool wxType::IsOk() const
114 // NB: m_strType can contain the :: operator; think to e.g. the
115 // "reverse_iterator_impl<wxString::const_iterator>" type
116 // It can also contain commas, * and & operators etc
118 return !m_strTypeClean
.IsEmpty();
121 bool wxType::operator==(const wxType
& m
) const
123 // brain-dead comparison:
125 if (m_strTypeClean
== m
.m_strTypeClean
&&
126 IsConst() == m
.IsConst() &&
127 IsStatic() == m
.IsStatic() &&
128 IsPointer() == m
.IsPointer() &&
129 IsReference() == m
.IsReference())
134 LogMessage("Type '%s' does not match type '%s'", m_strType
, m
.m_strType
);
135 LogMessage(" => TypeClean %s / %s; IsConst %d / %d; IsStatic %d / %d; IsPointer %d / %d; IsReference %d / %d",
136 m_strTypeClean
, m
.m_strTypeClean
, IsConst(), m
.IsConst(),
137 IsStatic(), m
.IsStatic(), IsPointer(), m
.IsPointer(),
138 IsReference(), m
.IsReference());
145 // ----------------------------------------------------------------------------
147 // ----------------------------------------------------------------------------
149 void wxArgumentType::SetDefaultValue(const wxString
& defval
, const wxString
& defvalForCmp
)
151 m_strDefaultValue
= defval
.Strip(wxString::both
);
152 m_strDefaultValueForCmp
= defvalForCmp
.IsEmpty() ?
153 m_strDefaultValue
: defvalForCmp
.Strip(wxString::both
);
156 // clean the default argument strings
157 // ----------------------------------
159 // Note: we adjust the aesthetic form of the m_strDefaultValue string for the "modify mode"
160 // of ifacecheck: we may need to write it out in an interface header
163 for (int i
=0; i
<2; i
++) // to avoid copying&pasting the code!
165 if (i
== 0) p
= &m_strDefaultValue
;
166 if (i
== 1) p
= &m_strDefaultValueForCmp
;
168 if (*p
== "0u") *p
= "0";
170 p
->Replace("0x000000001", "1");
171 p
->Replace("\\000\\000\\000", ""); // fix for unicode strings:
173 // ADHOC-FIX: for wxConv* default values
174 p
->Replace("wxConvAuto(wxFONTENCODING_DEFAULT)", "wxConvAuto()");
175 p
->Replace("wxGet_wxConvUTF8()", "wxConvUTF8");
176 p
->Replace("wxGet_wxConvLocal()", "wxConvLocal");
180 // clean ONLY the default argument string specific for comparison
181 // --------------------------------------------------------------
183 if (m_strDefaultValueForCmp
.StartsWith("wxT(") &&
184 m_strDefaultValueForCmp
.EndsWith(")"))
186 // get rid of the wxT() part
187 unsigned int len
= m_strDefaultValueForCmp
.Len();
188 m_strDefaultValueForCmp
= m_strDefaultValueForCmp
.Mid(4,len
-5);
192 // doxygen likes to put wxDateTime:: in front of all wxDateTime enums;
193 // fix this to avoid false positives
194 m_strDefaultValueForCmp
.Replace("wxDateTime::", "");
195 m_strDefaultValueForCmp
.Replace("wxStockGDI::", ""); // same story for some other classes
196 m_strDefaultValueForCmp
.Replace("wxHelpEvent::", ""); // same story for some other classes
197 m_strDefaultValueForCmp
.Replace("* GetColour(COLOUR_BLACK)", "*wxBLACK");
200 if (m_strDefaultValueForCmp
.Contains("wxGetTranslation"))
201 m_strDefaultValueForCmp
= "_(TOFIX)"; // TODO: wxGetTranslation gives problems to gccxml
204 bool wxArgumentType::operator==(const wxArgumentType
& m
) const
206 if ((const wxType
&)(*this) != (const wxType
&)m
)
210 // default values for style attributes of wxWindow-derived classes in gccxml appear as raw
211 // numbers; avoid false positives in this case!
212 if (m_strArgName
== m
.m_strArgName
&& m_strArgName
== "style" &&
213 (m_strDefaultValueForCmp
.IsNumber() || m
.m_strDefaultValueForCmp
.IsNumber()))
216 // fix for default values which were replaced by gcc-xml with their numeric values
217 // (at this point we know that m_strTypeClean == m.m_strTypeClean):
218 if (m_strTypeClean
== "long" || m_strTypeClean
== "int")
220 if ((m_strDefaultValueForCmp
.IsNumber() && m
.m_strDefaultValueForCmp
.StartsWith("wx")) ||
221 (m
.m_strDefaultValueForCmp
.IsNumber() && m_strDefaultValueForCmp
.StartsWith("wx")))
224 LogMessage("Supposing '%s' default value to be the same of '%s'...",
225 m_strDefaultValueForCmp
, m
.m_strDefaultValueForCmp
);
231 if (m_strDefaultValueForCmp
!= m
.m_strDefaultValueForCmp
)
233 // maybe the default values are numbers.
234 // in this case gccXML gives as default values things like '-0x0000001' instead of just '-1'.
235 // To handle these cases, we try to convert the default value strings to numbers:
236 long def1val
, def2val
;
237 if (m_strDefaultValueForCmp
.ToLong(&def1val
, 0 /* auto-detect */) &&
238 m
.m_strDefaultValueForCmp
.ToLong(&def2val
, 0 /* auto-detect */))
240 if (def1val
== def2val
)
241 return true; // the default values match
245 LogMessage("Argument type '%s = %s' has different default value from '%s = %s'",
246 m_strType
, m_strDefaultValueForCmp
, m
.m_strType
, m
.m_strDefaultValueForCmp
);
250 // we deliberately avoid checks on the argument name
256 // ----------------------------------------------------------------------------
258 // ----------------------------------------------------------------------------
260 bool wxMethod::IsOk() const
262 // NOTE: m_retType can be a wxEmptyType, and means that this method
263 // is a ctor or a dtor.
264 if (!m_retType
.IsOk() && m_retType
!=wxEmptyType
) {
265 LogError("'%s' method has invalid return type: %s", m_retType
.GetAsString());
269 if (m_strName
.IsEmpty())
272 // a function can't be both const and static or virtual and static!
273 if ((m_bConst
&& m_bStatic
) || ((m_bVirtual
|| m_bPureVirtual
) && m_bStatic
)) {
274 LogError("'%s' method can't be both const/static or virtual/static", m_strName
);
278 wxASSERT(!m_bPureVirtual
|| (m_bPureVirtual
&& m_bVirtual
));
280 for (unsigned int i
=0; i
<m_args
.GetCount(); i
++)
281 if (!m_args
[i
].IsOk()) {
282 LogError("'%s' method has invalid %d-th argument type: %s",
283 m_strName
, i
+1, m_args
[i
].GetAsString());
287 // NB: the default value of the arguments can contain pretty much everything
288 // (think to e.g. wxPoint(3+4/2,0) or *wxBLACK or someClass<type>)
289 // so we don't do any test on their contents
290 if (m_args
.GetCount()>0)
292 bool previousArgHasDefault
= m_args
[0].HasDefaultValue();
293 for (unsigned int i
=1; i
<m_args
.GetCount(); i
++)
295 if (previousArgHasDefault
&& !m_args
[i
].HasDefaultValue()) {
296 LogError("'%s' method has %d-th argument which has no default value "
297 "(while the previous one had one!)",
302 previousArgHasDefault
= m_args
[i
].HasDefaultValue();
309 bool wxMethod::MatchesExceptForAttributes(const wxMethod
& m
) const
311 if (GetReturnType() != m
.GetReturnType() ||
312 GetName() != m
.GetName())
315 LogMessage("The method '%s' does not match method '%s'; different names/rettype", GetName(), m
.GetName());
319 if (m_args
.GetCount()!=m
.m_args
.GetCount()) {
321 LogMessage("Method '%s' has %d arguments while '%s' has %d arguments",
322 m_strName
, m_args
.GetCount(), m_strName
, m
.m_args
.GetCount());
326 // compare argument types
327 for (unsigned int i
=0; i
<m_args
.GetCount(); i
++)
328 if (m_args
[i
] != m
.m_args
[i
])
334 bool wxMethod::ActsAsDefaultCtor() const
339 for (unsigned int i
=0; i
<m_args
.GetCount(); i
++)
340 if (!m_args
[i
].HasDefaultValue())
346 bool wxMethod::operator==(const wxMethod
& m
) const
349 if (IsConst() != m
.IsConst() ||
350 IsStatic() != m
.IsStatic() ||
351 IsVirtual() != m
.IsVirtual() ||
352 IsPureVirtual() != m
.IsPureVirtual() ||
353 IsDeprecated() != m
.IsDeprecated() ||
354 GetAccessSpecifier() != m
.GetAccessSpecifier())
357 LogMessage("The method '%s' does not match method '%s'; different attributes", GetName(), m
.GetName());
362 // check everything else
363 return MatchesExceptForAttributes(m
);
366 wxString
wxMethod::GetAsString(bool bWithArgumentNames
, bool bCleanDefaultValues
,
367 bool bDeprecated
, bool bAccessSpec
) const
371 // NOTE: for return and argument types, never use wxType::GetAsCleanString
372 // since in that way we'd miss important decorators like &,*,const etc
374 if (m_retType
!=wxEmptyType
)
375 ret
+= m_retType
.GetAsString() + " ";
376 //else; this is a ctor or dtor
378 ret
+= m_strName
+ "(";
380 for (unsigned int i
=0; i
<m_args
.GetCount(); i
++)
382 ret
+= m_args
[i
].GetAsString();
384 const wxString
& name
= m_args
[i
].GetArgumentName();
385 if (bWithArgumentNames
&& !name
.IsEmpty())
388 const wxString
& def
= bCleanDefaultValues
?
389 m_args
[i
].GetDefaultCleanValue() : m_args
[i
].GetDefaultValue();
396 if (m_args
.GetCount()>0)
397 ret
= ret
.Left(ret
.Len()-2);
404 ret
= "static " + ret
;
405 if (m_bVirtual
|| m_bPureVirtual
)
406 ret
= "virtual " + ret
;
409 if (m_bDeprecated
&& bDeprecated
)
410 ret
+= " [deprecated]";
419 case wxMAS_PROTECTED
:
420 ret
+= " [protected]";
431 void wxMethod::Dump(wxTextOutputStream
& stream
) const
433 stream
<< "[" + m_retType
.GetAsString() + "]";
434 stream
<< "[" + m_strName
+ "]";
436 for (unsigned int i
=0; i
<m_args
.GetCount(); i
++)
437 stream
<< "[" + m_args
[i
].GetAsString() + " " + m_args
[i
].GetArgumentName() +
438 "=" + m_args
[i
].GetDefaultValue() + "]";
445 stream
<< " VIRTUAL";
447 stream
<< " PURE-VIRTUAL";
449 stream
<< " DEPRECATED";
454 // ----------------------------------------------------------------------------
456 // ----------------------------------------------------------------------------
458 wxString
wxClass::GetNameWithoutTemplate() const
460 // NB: I'm not sure this is the right terminology for this function!
462 if (m_strName
.Contains("<"))
463 return m_strName
.Left(m_strName
.Find("<"));
467 bool wxClass::IsValidCtorForThisClass(const wxMethod
& m
) const
469 // remember that e.g. the ctor for wxWritableCharTypeBuffer<wchar_t> is
470 // named wxWritableCharTypeBuffer, without the <...> part!
472 if (m
.IsCtor() && m
.GetName() == GetNameWithoutTemplate())
478 bool wxClass::IsValidDtorForThisClass(const wxMethod
& m
) const
480 if (m
.IsDtor() && m
.GetName() == "~" + GetNameWithoutTemplate())
486 void wxClass::Dump(wxTextOutputStream
& out
) const
488 out
<< m_strName
+ "\n";
490 for (unsigned int i
=0; i
<m_methods
.GetCount(); i
++) {
492 // dump all our methods
494 m_methods
[i
].Dump(out
);
501 bool wxClass::CheckConsistency() const
503 for (unsigned int i
=0; i
<m_methods
.GetCount(); i
++)
504 for (unsigned int j
=0; j
<m_methods
.GetCount(); j
++)
505 if (i
!=j
&& m_methods
[i
] == m_methods
[j
])
507 LogError("class %s has two methods with the same prototype: '%s'",
508 m_strName
, m_methods
[i
].GetAsString());
512 //((wxClass*)this)->m_methods.RemoveAt(j);
519 const wxMethod
* wxClass::FindMethod(const wxMethod
& m
) const
521 for (unsigned int i
=0; i
<m_methods
.GetCount(); i
++)
522 if (m_methods
[i
] == m
)
523 return &m_methods
[i
];
527 const wxMethod
* wxClass::RecursiveUpwardFindMethod(const wxMethod
& m
,
528 const wxXmlInterface
* allclasses
) const
530 // first, search into *this
531 const wxMethod
* ret
= FindMethod(m
);
535 // then, search into its parents
536 for (unsigned int i
=0; i
<m_parents
.GetCount(); i
++)
538 // ignore non-wx-classes parents
539 // AD-HOC FIX: discard wxScrolledT_Helper parent as it always gives errors
540 if (m_parents
[i
].StartsWith("wx") && m_parents
[i
] != "wxScrolledT_Helper")
542 const wxClass
*parent
= allclasses
->FindClass(m_parents
[i
]);
544 LogError("Could not find parent '%s' of class '%s'...",
545 m_parents
[i
], GetName());
549 const wxMethod
*parentMethod
= parent
->RecursiveUpwardFindMethod(m
, allclasses
);
555 // could not find anything even in parent classes...
559 wxMethodPtrArray
wxClass::FindMethodsNamed(const wxString
& name
) const
561 wxMethodPtrArray ret
;
563 for (unsigned int i
=0; i
<m_methods
.GetCount(); i
++)
564 if (m_methods
[i
].GetName() == name
)
565 ret
.Add(&m_methods
[i
]);
571 wxMethodPtrArray
wxClass::RecursiveUpwardFindMethodsNamed(const wxString
& name
,
572 const wxXmlInterface
* allclasses
) const
574 // first, search into *this
575 wxMethodPtrArray ret
= FindMethodsNamed(name
);
576 if (ret
.GetCount()>0)
577 return ret
; // stop here, don't look upward in the parents
579 // then, search into parents of this class
580 for (unsigned int i
=0; i
<m_parents
.GetCount(); i
++)
582 // AD-HOC FIX: discard wxScrolledT_Helper parent as it always gives errors
583 if (m_parents
[i
].StartsWith("wx") && m_parents
[i
] != "wxScrolledT_Helper")
585 const wxClass
*parent
= allclasses
->FindClass(m_parents
[i
]);
587 LogError("Could not find parent '%s' of class '%s'...",
588 m_parents
[i
], GetName());
592 wxMethodPtrArray temp
= parent
->RecursiveUpwardFindMethodsNamed(name
, allclasses
);
593 WX_APPEND_ARRAY(ret
, temp
);
602 // ----------------------------------------------------------------------------
604 // ----------------------------------------------------------------------------
606 WX_DEFINE_SORTED_ARRAY(wxClass
*, wxSortedClassArray
);
608 int CompareWxClassObjects(wxClass
*item1
, wxClass
*item2
)
610 // sort alphabetically
611 return item1
->GetName().Cmp(item2
->GetName());
614 void wxXmlInterface::Dump(const wxString
& filename
)
616 wxFFileOutputStream
apioutput( filename
);
617 wxTextOutputStream
apiout( apioutput
);
619 // dump the classes in alphabetical order
620 wxSortedClassArray
sorted(CompareWxClassObjects
);
621 sorted
.Alloc(m_classes
.GetCount());
622 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
623 sorted
.Add(&m_classes
[i
]);
625 // now they have been sorted
626 for (unsigned int i
=0; i
<sorted
.GetCount(); i
++)
627 sorted
[i
]->Dump(apiout
);
630 bool wxXmlInterface::CheckParseResults() const
632 // this check can be quite slow, so do it only for debug releases:
634 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
635 if (!m_classes
[i
].CheckConsistency())
642 wxClassPtrArray
wxXmlInterface::FindClassesDefinedIn(const wxString
& headerfile
) const
646 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
647 if (m_classes
[i
].GetHeader() == headerfile
)
648 ret
.Add(&m_classes
[i
]);
654 // ----------------------------------------------------------------------------
655 // wxXmlGccInterface helper declarations
656 // ----------------------------------------------------------------------------
658 // or-able flags for a toResolveTypeItem->attrib:
659 #define ATTRIB_CONST 1
660 #define ATTRIB_REFERENCE 2
661 #define ATTRIB_POINTER 4
662 #define ATTRIB_ARRAY 8
664 // it may sound strange but gccxml, in order to produce shorter ID names
665 // uses (after the underscore) characters in range 0-9 and a-z in the ID names;
666 // in order to be able to translate such strings into numbers using strtoul()
667 // we use as base 10 (possible digits) + 25 (possible characters) = 35
668 #define GCCXML_BASE 35
670 class toResolveTypeItem
673 toResolveTypeItem() { attribs
=0; }
674 toResolveTypeItem(unsigned int refID
, unsigned int attribint
)
675 : ref(refID
), attribs(attribint
) {}
677 unsigned long ref
, // the referenced type's ID
678 attribs
; // the attributes of this reference
683 // for wxToResolveTypeHashMap, keys == gccXML IDs and values == toResolveTypeItem
684 WX_DECLARE_HASH_MAP( unsigned long, toResolveTypeItem
,
685 wxIntegerHash
, wxIntegerEqual
,
686 wxToResolveTypeHashMap
);
688 // for wxClassMemberIdHashMap, keys == gccXML IDs and values == wxClass which owns that member ID
689 WX_DECLARE_HASH_MAP( unsigned long, wxClass
*,
690 wxIntegerHash
, wxIntegerEqual
,
691 wxClassMemberIdHashMap
);
695 typedef std::map
<unsigned long, toResolveTypeItem
> wxToResolveTypeHashMap
;
699 // utility to parse gccXML ID values;
700 // this function is equivalent to wxString(str).Mid(1).ToULong(&id, GCCXML_BASE)
701 // but is a little bit faster
702 bool getID(unsigned long *id
, const wxString
& str
)
704 const wxStringCharType
* const start
= str
.wx_str()+1;
705 wxStringCharType
*end
;
706 #if wxUSE_UNICODE_WCHAR
707 unsigned long val
= wcstoul(start
, &end
, GCCXML_BASE
);
709 unsigned long val
= strtoul(start
, &end
, GCCXML_BASE
);
712 // return true only if scan was stopped by the terminating NUL and
713 // if the string was not empty to start with and no under/overflow
715 if ( *end
!= '\0' || end
== start
|| errno
== ERANGE
|| errno
== EINVAL
)
722 // utility specialized to parse efficiently the gccXML list of IDs which occur
723 // in nodes like <Class> ones... i.e. numeric values separed by " _" token
724 bool getMemberIDs(wxClassMemberIdHashMap
* map
, wxClass
* p
, const wxString
& str
)
726 const wxStringCharType
* const start
= str
.wx_str();
727 #if wxUSE_UNICODE_WCHAR
728 size_t len
= wcslen(start
);
730 size_t len
= strlen(start
);
733 if (len
== 0 || start
[0] != '_')
736 const wxStringCharType
*curpos
= start
,
738 wxStringCharType
*nexttoken
;
742 // curpos always points to the underscore of the next token to parse:
743 #if wxUSE_UNICODE_WCHAR
744 unsigned long id
= wcstoul(curpos
+1, &nexttoken
, GCCXML_BASE
);
746 unsigned long id
= strtoul(curpos
+1, &nexttoken
, GCCXML_BASE
);
748 if ( *nexttoken
!= ' ' || errno
== ERANGE
|| errno
== EINVAL
)
751 // advance current position
752 curpos
= nexttoken
+ 1;
754 // add this ID to the hashmap
755 wxClassMemberIdHashMap::value_type
v(id
, p
);
763 // ----------------------------------------------------------------------------
765 // ----------------------------------------------------------------------------
767 bool wxXmlGccInterface::Parse(const wxString
& filename
)
773 LogMessage("Parsing %s...", filename
);
775 if (!doc
.Load(filename
)) {
776 LogError("can't load %s", filename
);
780 // start processing the XML file
781 if (doc
.GetRoot()->GetName() != "GCC_XML") {
782 LogError("invalid root node for %s", filename
);
786 wxString version
= doc
.GetRoot()->GetAttribute("cvs_revision");
789 #define MIN_REVISION 120
791 if (!version
.StartsWith("1."))
795 unsigned long rev
= 0;
796 if (!version
.Mid(2).ToULong(&rev
))
799 if (rev
< MIN_REVISION
)
805 LogError("The version of GCC-XML used for the creation of %s is too old; "
806 "the cvs_revision attribute of the root node reports '%s', "
807 "minimal required is 1.%d.", filename
, version
, MIN_REVISION
);
811 wxToResolveTypeHashMap toResolveTypes
;
812 wxClassMemberIdHashMap members
;
813 wxTypeIdHashMap types
;
814 wxTypeIdHashMap files
;
815 wxTypeIdHashMap typedefs
;
817 // prealloc quite a lot of memory!
818 m_classes
.Alloc(ESTIMATED_NUM_CLASSES
);
820 // build a list of wx classes and in general of all existent types
821 child
= doc
.GetRoot()->GetChildren();
824 const wxString
& n
= child
->GetName();
826 unsigned long id
= 0;
827 if (!getID(&id
, child
->GetAttribute("id")) || (id
== 0 && n
!= "File")) {
829 // NOTE: <File> nodes can have an id == "f0"...
831 LogError("Invalid id for node %s: %s", n
, child
->GetAttribute("id"));
837 wxString cname
= child
->GetAttribute("name");
838 if (cname
.IsEmpty()) {
839 LogError("Invalid empty name for '%s' node", n
);
843 // only register wx classes (do remember also the IDs of their members)
844 if (cname
.StartsWith("wx"))
846 // NB: "file" attribute contains an ID value that we'll resolve later
847 m_classes
.Add(wxClass(cname
, child
->GetAttribute("file")));
849 // the just-inserted class:
850 wxClass
*newClass
= &m_classes
.Last();
852 // now get a list of the base classes:
853 wxXmlNode
*baseNode
= child
->GetChildren();
856 // for now we store as "parents" only the parent IDs...
857 // later we will resolve them into full class names
858 if (baseNode
->GetName() == "Base")
859 newClass
->AddParent(baseNode
->GetAttribute("type"));
861 baseNode
= baseNode
->GetNext();
864 const wxString
& ids
= child
->GetAttribute("members");
867 if (child
->GetAttribute("incomplete") != "1") {
868 LogError("Invalid member IDs for '%s' class node: %s",
869 cname
, child
->GetAttribute("id"));
872 //else: don't warn the user; it looks like "incomplete" classes
873 // never have any member...
877 // decode the non-empty list of IDs:
878 if (!getMemberIDs(&members
, newClass
, ids
)) {
879 LogError("Invalid member IDs for '%s' class node: %s",
880 cname
, child
->GetAttribute("id"));
886 // register this class also as possible return/argument type:
889 else if (n
== "Typedef")
891 unsigned long typeId
= 0;
892 if (!getID(&typeId
, child
->GetAttribute("type"))) {
893 LogError("Invalid type for node %s: %s", n
, child
->GetAttribute("type"));
897 // this typedef node tell us that every type referenced with the
898 // "typeId" ID should be called with another name:
899 wxString name
= child
->GetAttribute("name");
901 // save this typedef in a separate hashmap...
902 typedefs
[typeId
] = name
;
906 else if (n
== "PointerType" || n
== "ReferenceType" ||
907 n
== "CvQualifiedType" || n
== "ArrayType")
909 unsigned long type
= 0;
910 if (!getID(&type
, child
->GetAttribute("type")) || type
== 0) {
911 LogError("Invalid type for node %s: %s", n
, child
->GetAttribute("type"));
915 unsigned long attr
= 0;
916 if (n
== "PointerType")
917 attr
= ATTRIB_POINTER
;
918 else if (n
== "ReferenceType")
919 attr
= ATTRIB_REFERENCE
;
920 else if (n
== "CvQualifiedType" && child
->GetAttribute("const") == "1")
922 else if (n
== "ArrayType")
925 // these nodes make reference to other types... we'll resolve them later
926 toResolveTypes
[id
] = toResolveTypeItem(type
, attr
);
928 else if (n
== "FunctionType" || n
== "MethodType")
931 TODO: parsing FunctionType and MethodType nodes is not as easy
932 as for other "simple" types.
936 wxXmlNode
*arg
= child
->GetChildren();
939 if (arg
->GetName() == "Argument")
940 argstr
+= arg
->GetAttribute("type") + ", ";
941 arg
= arg
->GetNext();
944 if (argstr
.Len() > 0)
945 argstr
= argstr
.Left(argstr
.Len()-2); // remove final comma
947 // these nodes make reference to other types... we'll resolve them later
948 //toResolveTypes[id] = toResolveTypeItem(ret, 0);
949 //types[id] = child->GetAttribute("returns") + "(" + argstr + ")";
951 types
[id
] = "TOFIX"; // typically this type will be "fixed" thanks
952 // to a typedef later...
954 else if (n
== "File")
956 if (!child
->GetAttribute("id").StartsWith("f")) {
957 LogError("Unexpected file ID: %s", child
->GetAttribute("id"));
961 // just ignore this node... all file IDs/names were already parsed
962 files
[id
] = child
->GetAttribute("name");
966 // we register everything else as a possible return/argument type:
967 const wxString
& name
= child
->GetAttribute("name");
972 //typeNames.Add(name);
977 // this may happen with unnamed structs/union, special ctors,
978 // or other exotic things which we are not interested to, since
979 // they're never used as return/argument types by wxWidgets methods
982 LogWarning("Type node '%s' with ID '%s' does not have name attribute",
983 n
, child
->GetAttribute("id"));
989 child
= child
->GetNext();
991 // give feedback to the user about the progress...
992 if ((++nodes%PROGRESS_RATE
)==0) ShowProgress();
995 // some nodes with IDs referenced by methods as return/argument types, do reference
996 // in turn other nodes (see PointerType, ReferenceType and CvQualifierType above);
997 // thus we need to resolve their name iteratively:
998 while (toResolveTypes
.size()>0)
1001 LogMessage("%d types were collected; %d types need yet to be resolved...",
1002 types
.size(), toResolveTypes
.size());
1004 for (wxToResolveTypeHashMap::iterator i
= toResolveTypes
.begin();
1005 i
!= toResolveTypes
.end();)
1007 unsigned long id
= i
->first
;
1008 unsigned long referenced
= i
->second
.ref
;
1010 wxTypeIdHashMap::iterator primary
= types
.find(referenced
);
1011 if (primary
!= types
.end())
1013 // this to-resolve-type references a "primary" type
1015 wxString newtype
= primary
->second
;
1016 int attribs
= i
->second
.attribs
;
1018 // attribs may contain a combination of ATTRIB_* flags:
1019 if (attribs
& ATTRIB_CONST
)
1020 newtype
= "const " + newtype
;
1021 if (attribs
& ATTRIB_REFERENCE
)
1022 newtype
= newtype
+ "&";
1023 if (attribs
& ATTRIB_POINTER
)
1024 newtype
= newtype
+ "*";
1025 if (attribs
& ATTRIB_ARRAY
)
1026 newtype
= newtype
+ "[]";
1028 // add the resolved type to the list of "primary" types
1029 if (newtype
.Contains("TOFIX") && typedefs
[id
] != "")
1030 types
[id
] = typedefs
[id
]; // better use a typedef for this type!
1032 types
[id
] = newtype
;
1034 // this one has been resolved; erase it through its iterator!
1035 toResolveTypes
.erase(i
);
1037 // now iterator i is invalid; assign it again to the beginning
1038 i
= toResolveTypes
.begin();
1042 // then search in the referenced types themselves:
1043 wxToResolveTypeHashMap::iterator idx2
= toResolveTypes
.find(referenced
);
1044 if (idx2
!= toResolveTypes
.end())
1046 // merge this to-resolve-type with the idx2->second type
1047 i
->second
.ref
= idx2
->second
.ref
;
1048 i
->second
.attribs
|= idx2
->second
.attribs
;
1050 // this type will eventually be solved in the next while() iteration
1055 LogError("Cannot solve '%d' reference type!", referenced
);
1062 // resolve header names
1063 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
1065 unsigned long fileID
= 0;
1066 if (!getID(&fileID
, m_classes
[i
].GetHeader()) || fileID
== 0) {
1067 LogError("invalid header id: %s", m_classes
[i
].GetHeader());
1072 wxTypeIdHashMap::const_iterator idx
= files
.find(fileID
);
1073 if (idx
== files
.end())
1075 // this is an error!
1076 LogError("couldn't find file ID '%s'", m_classes
[i
].GetHeader());
1079 m_classes
[i
].SetHeader(idx
->second
);
1082 // resolve parent names
1083 for (unsigned int i
=0; i
<m_classes
.GetCount(); i
++)
1085 for (unsigned int k
=0; k
<m_classes
[i
].GetParentCount(); k
++)
1089 if (!getID(&id
, m_classes
[i
].GetParent(k
))) {
1090 LogError("invalid parent class ID for '%s'", m_classes
[i
].GetName());
1094 wxTypeIdHashMap::const_iterator idx
= types
.find(id
);
1095 if (idx
== types
.end())
1097 // this is an error!
1098 LogError("couldn't find parent class ID '%d'", id
);
1101 // replace k-th parent with its true name:
1102 m_classes
[i
].SetParent(k
, idx
->second
);
1106 // build the list of the wx methods
1107 child
= doc
.GetRoot()->GetChildren();
1110 wxString n
= child
->GetName(), acc
= child
->GetAttribute("access");
1112 // only register public&protected methods
1113 if ((acc
== "public" || acc
== "protected") &&
1114 (n
== "Method" || n
== "Constructor" || n
== "Destructor" || n
== "OperatorMethod"))
1116 unsigned long id
= 0;
1117 if (!getID(&id
, child
->GetAttribute("id"))) {
1118 LogError("invalid ID for node '%s' with ID '%s'", n
, child
->GetAttribute("id"));
1122 wxClassMemberIdHashMap::const_iterator it
= members
.find(id
);
1123 if (it
!= members
.end())
1125 wxClass
*p
= it
->second
;
1127 // this <Method> node is a method of the i-th class!
1129 if (!ParseMethod(child
, types
, newfunc
)) {
1130 LogError("The method '%s' could not be added to class '%s'",
1131 child
->GetAttribute("demangled"), p
->GetName());
1135 // do some additional check that we can do only here:
1137 if (newfunc
.IsCtor() && !p
->IsValidCtorForThisClass(newfunc
)) {
1138 LogError("The method '%s' does not seem to be a ctor for '%s'",
1139 newfunc
.GetName(), p
->GetName());
1142 if (newfunc
.IsDtor() && !p
->IsValidDtorForThisClass(newfunc
)) {
1143 LogError("The method '%s' does not seem to be a dtor for '%s'",
1144 newfunc
.GetName(), p
->GetName());
1148 p
->AddMethod(newfunc
);
1152 child
= child
->GetNext();
1154 // give feedback to the user about the progress...
1155 if ((++nodes%PROGRESS_RATE
)==0) ShowProgress();
1158 if (!CheckParseResults())
1164 bool wxXmlGccInterface::ParseMethod(const wxXmlNode
*p
,
1165 const wxTypeIdHashMap
& types
,
1168 // get the real name
1169 wxString name
= p
->GetAttribute("name").Strip(wxString::both
);
1170 if (p
->GetName() == "Destructor")
1172 else if (p
->GetName() == "OperatorMethod")
1173 name
= "operator" + name
;
1175 // resolve return type
1177 unsigned long retid
= 0;
1178 if (!getID(&retid
, p
->GetAttribute("returns")) || retid
== 0)
1180 if (p
->GetName() != "Destructor" && p
->GetName() != "Constructor") {
1181 LogError("Empty return ID for method '%s', with ID '%s'",
1182 name
, p
->GetAttribute("id"));
1188 wxTypeIdHashMap::const_iterator retidx
= types
.find(retid
);
1189 if (retidx
== types
.end()) {
1190 LogError("Could not find return type ID '%s'", retid
);
1194 ret
= wxType(retidx
->second
);
1196 LogError("Invalid return type '%s' for method '%s', with ID '%s'",
1197 retidx
->second
, name
, p
->GetAttribute("id"));
1202 // resolve argument types
1203 wxArgumentTypeArray argtypes
;
1204 wxXmlNode
*arg
= p
->GetChildren();
1207 if (arg
->GetName() == "Argument")
1209 unsigned long id
= 0;
1210 if (!getID(&id
, arg
->GetAttribute("type")) || id
== 0) {
1211 LogError("Invalid argument type ID '%s' for method '%s' with ID %s",
1212 arg
->GetAttribute("type"), name
, p
->GetAttribute("id"));
1216 wxTypeIdHashMap::const_iterator idx
= types
.find(id
);
1217 if (idx
== types
.end()) {
1218 LogError("Could not find argument type ID '%s'", id
);
1222 argtypes
.Add(wxArgumentType(idx
->second
,
1223 arg
->GetAttribute("default"),
1224 arg
->GetAttribute("name")));
1227 arg
= arg
->GetNext();
1230 m
.SetReturnType(ret
);
1232 m
.SetArgumentTypes(argtypes
);
1233 m
.SetConst(p
->GetAttribute("const") == "1");
1234 m
.SetStatic(p
->GetAttribute("static") == "1");
1236 // NOTE: gccxml is smart enough to mark as virtual those functions
1237 // which are declared virtual in base classes but don't have
1238 // the "virtual" keyword explicitely indicated in the derived
1239 // classes... so we don't need any further logic for virtuals
1241 m
.SetVirtual(p
->GetAttribute("virtual") == "1");
1242 m
.SetPureVirtual(p
->GetAttribute("pure_virtual") == "1");
1243 m
.SetDeprecated(p
->GetAttribute("attributes") == "deprecated");
1245 // decode access specifier
1246 if (p
->GetAttribute("access") == "public")
1247 m
.SetAccessSpecifier(wxMAS_PUBLIC
);
1248 else if (p
->GetAttribute("access") == "protected")
1249 m
.SetAccessSpecifier(wxMAS_PROTECTED
);
1250 else if (p
->GetAttribute("access") == "private")
1251 m
.SetAccessSpecifier(wxMAS_PRIVATE
);
1254 LogError("The prototype '%s' is not valid!", m
.GetAsString());
1263 // ----------------------------------------------------------------------------
1264 // wxXmlDoxygenInterface global helpers
1265 // ----------------------------------------------------------------------------
1267 static wxString
GetTextFromChildren(const wxXmlNode
*n
)
1271 // consider the tree
1273 // <a><b>this</b> is a <b>string</b></a>
1282 // unlike wxXmlNode::GetNodeContent() which would return " is a "
1283 // this function returns "this is a string"
1285 wxXmlNode
*ref
= n
->GetChildren();
1287 if (ref
->GetType() == wxXML_ELEMENT_NODE
)
1288 text
+= ref
->GetNodeContent();
1289 else if (ref
->GetType() == wxXML_TEXT_NODE
)
1290 text
+= ref
->GetContent();
1292 LogWarning("Unexpected node type while getting text from '%s' node", n
->GetName());
1294 ref
= ref
->GetNext();
1300 static bool HasTextNodeContaining(const wxXmlNode
*parent
, const wxString
& name
)
1305 wxXmlNode
*p
= parent
->GetChildren();
1308 switch (p
->GetType())
1310 case wxXML_TEXT_NODE
:
1311 if (p
->GetContent() == name
)
1315 case wxXML_ELEMENT_NODE
:
1316 // recurse into this node...
1317 if (HasTextNodeContaining(p
, name
))
1332 static const wxXmlNode
* FindNodeNamed(const wxXmlNode
* parent
, const wxString
& name
)
1337 const wxXmlNode
*p
= parent
->GetChildren();
1340 if (p
->GetName() == name
)
1343 // search recursively in the children of this node
1344 const wxXmlNode
*ret
= FindNodeNamed(p
, name
);
1354 int GetAvailabilityFor(const wxXmlNode
*node
)
1356 // identify <onlyfor> custom XML tags
1357 const wxXmlNode
* onlyfor
= FindNodeNamed(node
, "onlyfor");
1359 return wxPORT_UNKNOWN
;
1361 wxArrayString ports
= wxSplit(onlyfor
->GetNodeContent(), ',');
1362 int nAvail
= wxPORT_UNKNOWN
;
1363 for (unsigned int i
=0; i
< ports
.GetCount(); i
++)
1365 if (!ports
[i
].StartsWith("wx")) {
1366 LogError("unexpected port ID '%s'", ports
[i
]);
1370 nAvail
|= wxPlatformInfo::GetPortId(ports
[i
].Mid(2));
1377 // ----------------------------------------------------------------------------
1378 // wxXmlDoxygenInterface
1379 // ----------------------------------------------------------------------------
1381 bool wxXmlDoxygenInterface::Parse(const wxString
& filename
)
1383 wxXmlDocument index
;
1384 wxXmlNode
*compound
;
1386 LogMessage("Parsing %s...", filename
);
1388 if (!index
.Load(filename
)) {
1389 LogError("can't load %s", filename
);
1393 // start processing the index:
1394 if (index
.GetRoot()->GetName() != "doxygenindex") {
1395 LogError("invalid root node for %s", filename
);
1400 NB: we may need in future to do a version-check here if the
1401 format of the XML generated by doxygen changes.
1402 For now (doxygen version 1.5.5), this check is not required
1403 since AFAIK the XML format never changed since it was introduced.
1406 m_classes
.Alloc(ESTIMATED_NUM_CLASSES
);
1408 // process files referenced by this index file
1409 compound
= index
.GetRoot()->GetChildren();
1412 if (compound
->GetName() == "compound" &&
1413 compound
->GetAttribute("kind") == "class")
1415 wxString refid
= compound
->GetAttribute("refid");
1417 wxFileName
fn(filename
);
1418 if (!ParseCompoundDefinition(fn
.GetPath(wxPATH_GET_SEPARATOR
) + refid
+ ".xml"))
1422 compound
= compound
->GetNext();
1426 if (!CheckParseResults())
1432 bool wxXmlDoxygenInterface::ParseCompoundDefinition(const wxString
& filename
)
1434 wxClassMemberIdHashMap parents
;
1440 LogMessage("Parsing %s...", filename
);
1442 if (!doc
.Load(filename
)) {
1443 LogError("can't load %s", filename
);
1447 // start processing this compound definition XML
1448 if (doc
.GetRoot()->GetName() != "doxygen") {
1449 LogError("invalid root node for %s", filename
);
1453 // build a list of wx classes
1454 child
= doc
.GetRoot()->GetChildren();
1457 if (child
->GetName() == "compounddef" &&
1458 child
->GetAttribute("kind") == "class")
1462 wxString absoluteFile
, header
;
1464 wxXmlNode
*subchild
= child
->GetChildren();
1467 wxString kind
= subchild
->GetAttribute("kind");
1469 // parse only public&protected functions:
1470 if (subchild
->GetName() == "sectiondef" &&
1471 (kind
== "public-func" || kind
== "protected-func"))
1474 wxXmlNode
*membernode
= subchild
->GetChildren();
1477 if (membernode
->GetName() == "memberdef" &&
1478 membernode
->GetAttribute("kind") == "function")
1482 if (!ParseMethod(membernode
, m
, header
)) {
1483 LogError("The method '%s' could not be added to class '%s'",
1484 m
.GetName(), klass
.GetName());
1488 if (kind
== "public-func")
1489 m
.SetAccessSpecifier(wxMAS_PUBLIC
);
1490 else if (kind
== "protected-func")
1491 m
.SetAccessSpecifier(wxMAS_PROTECTED
);
1492 else if (kind
== "private-func")
1493 m
.SetAccessSpecifier(wxMAS_PRIVATE
);
1495 if (absoluteFile
.IsEmpty())
1496 absoluteFile
= header
;
1497 else if (header
!= absoluteFile
)
1499 LogError("The method '%s' is documented in a different "
1500 "file from others (which belong to '%s') ?",
1501 header
, absoluteFile
);
1508 membernode
= membernode
->GetNext();
1511 // all methods of this class were taken from the header "absoluteFile":
1512 klass
.SetHeader(absoluteFile
);
1514 else if (subchild
->GetName() == "compoundname")
1516 klass
.SetName(subchild
->GetNodeContent());
1518 /*else if (subchild->GetName() == "includes")
1520 // NOTE: we'll get the header from the <location> tags
1521 // scattered inside <memberdef> tags instead of
1522 // this <includes> tag since it does not contain
1523 // the absolute path of the header
1525 klass.SetHeader(subchild->GetNodeContent());
1527 else if (subchild
->GetName() == "detaileddescription")
1529 // identify <onlyfor> custom XML tags
1530 klass
.SetAvailability(GetAvailabilityFor(subchild
));
1532 else if (subchild
->GetName() == "basecompoundref")
1534 // add the name of this parent to the list of klass' parents
1535 klass
.AddParent(subchild
->GetNodeContent());
1538 subchild
= subchild
->GetNext();
1543 m_classes
.Add(klass
);
1545 LogWarning("discarding class '%s' with %d methods...",
1546 klass
.GetName(), klass
.GetMethodCount());
1549 child
= child
->GetNext();
1551 // give feedback to the user about the progress...
1552 if ((++nodes%PROGRESS_RATE
)==0) ShowProgress();
1558 bool wxXmlDoxygenInterface::ParseMethod(const wxXmlNode
* p
, wxMethod
& m
, wxString
& header
)
1560 wxArgumentTypeArray args
;
1563 wxXmlNode
*child
= p
->GetChildren();
1566 if (child
->GetName() == "name")
1567 m
.SetName(child
->GetNodeContent());
1568 else if (child
->GetName() == "type")
1569 m
.SetReturnType(wxType(GetTextFromChildren(child
)));
1570 else if (child
->GetName() == "param")
1572 wxString typestr
, namestr
, defstr
, arrstr
;
1573 wxXmlNode
*n
= child
->GetChildren();
1576 if (n
->GetName() == "type")
1577 // if the <type> node has children, they should be all TEXT and <ref> nodes
1578 // and we need to take the text they contain, in the order they appear
1579 typestr
= GetTextFromChildren(n
);
1580 else if (n
->GetName() == "declname")
1581 namestr
= GetTextFromChildren(n
);
1582 else if (n
->GetName() == "defval")
1583 defstr
= GetTextFromChildren(n
).Strip(wxString::both
);
1584 else if (n
->GetName() == "array")
1585 arrstr
= GetTextFromChildren(n
);
1590 if (typestr
.IsEmpty()) {
1591 LogError("cannot find type node for a param in method '%s'", m
.GetName());
1595 wxArgumentType
newarg(typestr
+ arrstr
, defstr
, namestr
);
1597 // can we use preprocessor output to transform the default value
1598 // into the same form which gets processed by wxXmlGccInterface?
1599 wxStringHashMap::const_iterator it
= m_preproc
.find(defstr
);
1600 if (it
!= m_preproc
.end())
1601 newarg
.SetDefaultValue(defstr
, it
->second
);
1605 else if (child
->GetName() == "location")
1608 if (child
->GetAttribute("line").ToLong(&line
))
1609 m
.SetLocation((int)line
);
1610 header
= child
->GetAttribute("file");
1612 else if (child
->GetName() == "detaileddescription")
1614 // when a method has a @deprecated tag inside its description,
1615 // Doxygen outputs somewhere nested inside <detaileddescription>
1616 // a <xreftitle>Deprecated</xreftitle> tag.
1617 m
.SetDeprecated(HasTextNodeContaining(child
, "Deprecated"));
1619 // identify <onlyfor> custom XML tags
1620 m
.SetAvailability(GetAvailabilityFor(child
));
1623 child
= child
->GetNext();
1626 m
.SetArgumentTypes(args
);
1627 m
.SetConst(p
->GetAttribute("const")=="yes");
1628 m
.SetStatic(p
->GetAttribute("static")=="yes");
1630 // NOTE: Doxygen is smart enough to mark as virtual those functions
1631 // which are declared virtual in base classes but don't have
1632 // the "virtual" keyword explicitely indicated in the derived
1633 // classes... so we don't need any further logic for virtuals
1635 m
.SetVirtual(p
->GetAttribute("virt")=="virtual");
1636 m
.SetPureVirtual(p
->GetAttribute("virt")=="pure-virtual");
1639 LogError("The prototype '%s' is not valid!", m
.GetAsString());