]> git.saurik.com Git - wxWidgets.git/blame - utils/ifacecheck/src/xmlparser.h
wxMessageBox off the main thread lost result code.
[wxWidgets.git] / utils / ifacecheck / src / xmlparser.h
CommitLineData
5934cda1
FM
1/////////////////////////////////////////////////////////////////////////////
2// Name: xmlparser.h
3// Purpose: Parser of the API/interface XML files
4// Author: Francesco Montorsi
5// Created: 2008/03/17
5934cda1
FM
6// Copyright: (c) 2008 Francesco Montorsi
7// Licence: wxWindows licence
8/////////////////////////////////////////////////////////////////////////////
9
10
11#ifndef _XMLPARSER_H_
12#define _XMLPARSER_H_
13
14#include <wx/txtstrm.h>
15#include <wx/dynarray.h>
16#include <wx/xml/xml.h>
d2367a0f 17#include <wx/platinfo.h>
7f8fae98 18#include <wx/log.h>
5934cda1 19
97f0dbd6
FM
20
21/*
22 IMPORTANT
23 =========
24
25 Any fix aimed to reduce "false positives" which involves
26 references to a specific wxWidgets class is marked in
27 ifacecheck sources with the string:
28
29 // ADHOC-FIX:
723de43b 30 // ...fix description...
97f0dbd6
FM
31*/
32
7f8fae98
FM
33// NOTE: all messages in this way are printed on the stderr
34//#define wxLogWarning wxLogMessage
97f0dbd6 35
5934cda1
FM
36
37// ----------------------------------------------------------------------------
38// Represents a type with or without const/static/ qualifier
39// and with or without & and * operators
40// ----------------------------------------------------------------------------
41class wxType
42{
43public:
44 wxType() {}
45 wxType(const wxString& type)
f270e1dd 46 { SetTypeFromString(type); }
5934cda1 47
f270e1dd 48 void SetTypeFromString(const wxString& t);
5934cda1
FM
49 wxString GetAsString() const
50 { return m_strType; }
2cb3a02f
FM
51
52 // returns this type _without_ any decoration,
53 // including the & (which indicates this is a reference),
54 // the * (which indicates this is a pointer), etc.
97f0dbd6
FM
55 wxString GetAsCleanString() const
56 { return m_strTypeClean; }
5934cda1 57
5934cda1
FM
58 bool IsConst() const
59 { return m_strType.Contains("const"); }
60 bool IsStatic() const
61 { return m_strType.Contains("static"); }
62 bool IsPointer() const
63 { return m_strType.Contains("*"); }
64 bool IsReference() const
65 { return m_strType.Contains("&"); }
66
67 bool operator==(const wxType& m) const;
68 bool operator!=(const wxType& m) const
69 { return !(*this == m); }
70
71 bool IsOk() const;
72
73protected:
658d9804
FM
74 wxString m_strType,
75 m_strTypeClean; // m_strType "cleaned" of its attributes
76 // (only for internal use)
5934cda1
FM
77};
78
79extern wxType wxEmptyType;
80WX_DECLARE_OBJARRAY(wxType, wxTypeArray);
81
82
f270e1dd
FM
83// ----------------------------------------------------------------------------
84// Represents a type used as argument for some wxMethod
85// ----------------------------------------------------------------------------
86class wxArgumentType : public wxType
87{
88public:
89 wxArgumentType() {}
90 wxArgumentType(const wxString& type, const wxString& defVal,
91 const wxString& argName = wxEmptyString)
92 { SetTypeFromString(type); SetDefaultValue(defVal); m_strArgName = argName; }
93
94 void SetArgumentName(const wxString& name)
95 { m_strArgName=name.Strip(wxString::both); }
96 wxString GetArgumentName() const
97 { return m_strArgName; }
98
2cb3a02f
FM
99 void SetDefaultValue(const wxString& defval,
100 const wxString& defvalForCmp = wxEmptyString);
f270e1dd
FM
101 wxString GetDefaultValue() const
102 { return m_strDefaultValue; }
2cb3a02f
FM
103
104 // returns the default value used for comparisons
97f0dbd6 105 wxString GetDefaultCleanValue() const
2cb3a02f 106 { return m_strDefaultValueForCmp; }
f270e1dd 107
203ba76a
FM
108 bool HasDefaultValue() const
109 { return !m_strDefaultValue.IsEmpty(); }
110
f270e1dd
FM
111 bool operator==(const wxArgumentType& m) const;
112 bool operator!=(const wxArgumentType& m) const
113 { return !(*this == m); }
114
115protected:
116 wxString m_strDefaultValue;
5570107a
FM
117
118 // this string may differ from m_strDefaultValue if there were
2cb3a02f
FM
119 // preprocessor substitutions or other "replacements" done to
120 // avoid false errors.
5570107a
FM
121 wxString m_strDefaultValueForCmp;
122
2cb3a02f
FM
123 // the argument name
124 wxString m_strArgName;
f270e1dd
FM
125};
126
127extern wxArgumentType wxEmptyArgumentType;
128WX_DECLARE_OBJARRAY(wxArgumentType, wxArgumentTypeArray);
129
5934cda1 130
8cd22478
FM
131enum wxMethodAccessSpecifier
132{
133 wxMAS_PUBLIC,
134 wxMAS_PROTECTED,
135 wxMAS_PRIVATE
136};
137
5934cda1
FM
138// ----------------------------------------------------------------------------
139// Represents a single prototype of a class' member.
140// ----------------------------------------------------------------------------
141class wxMethod
142{
143public:
0403e5dc 144 wxMethod()
03d4f7b9 145 { m_bConst=m_bVirtual=m_bPureVirtual=m_bStatic=m_bDeprecated=false;
8cd22478 146 m_nLine=-1; m_nAvailability=wxPORT_UNKNOWN; m_access=wxMAS_PUBLIC; }
0403e5dc 147
5934cda1 148 wxMethod(const wxType& rettype, const wxString& name,
f270e1dd 149 const wxArgumentTypeArray& arguments,
5934cda1
FM
150 bool isConst, bool isStatic, bool isVirtual)
151 : m_retType(rettype), m_strName(name.Strip(wxString::both)),
152 m_bConst(isConst), m_bStatic(isStatic), m_bVirtual(isVirtual)
f270e1dd 153 { SetArgumentTypes(arguments); m_nLine=-1; }
5934cda1
FM
154
155
156public: // getters
157
97f0dbd6 158 // bWithArgumentNames = output argument names?
8cd22478 159 // bCleanDefaultValues = output clean argument default values?
97f0dbd6 160 // bDeprecated = output [deprecated] next to deprecated methods?
8cd22478
FM
161 // bAccessSpec = output [public], [protected] or [private] next to method?
162 //
163 // TODO: convert to readable flags this set of bools
97f0dbd6 164 wxString GetAsString(bool bWithArgumentNames = true,
8cd22478
FM
165 bool bCleanDefaultValues = false,
166 bool bDeprecated = false,
167 bool bAccessSpec = false) const;
5934cda1
FM
168
169 // parser of the prototype:
170 // all these functions return strings with spaces stripped
171 wxType GetReturnType() const
172 { return m_retType; }
173 wxString GetName() const
174 { return m_strName; }
f3998820
FM
175 const wxArgumentTypeArray& GetArgumentTypes() const
176 { return m_args; }
177 wxArgumentTypeArray& GetArgumentTypes()
5934cda1 178 { return m_args; }
5934cda1
FM
179 int GetLocation() const
180 { return m_nLine; }
03d4f7b9
FM
181 int GetAvailability() const
182 { return m_nAvailability; }
8cd22478
FM
183 wxMethodAccessSpecifier GetAccessSpecifier() const
184 { return m_access; }
5934cda1
FM
185
186 bool IsConst() const
187 { return m_bConst; }
188 bool IsStatic() const
189 { return m_bStatic; }
190 bool IsVirtual() const
191 { return m_bVirtual; }
a7be99c8
FM
192 bool IsPureVirtual() const
193 { return m_bPureVirtual; }
5934cda1
FM
194
195 bool IsOk() const;
196 bool IsCtor() const
197 { return m_retType==wxEmptyType && !m_strName.StartsWith("~"); }
198 bool IsDtor() const
199 { return m_retType==wxEmptyType && m_strName.StartsWith("~"); }
83fdf796
FM
200 bool IsOperator() const
201 { return m_strName.StartsWith("operator"); }
5934cda1 202
0403e5dc
FM
203 bool IsDeprecated() const
204 { return m_bDeprecated; }
205
5934cda1 206
83fdf796 207
5934cda1
FM
208public: // setters
209
210 void SetReturnType(const wxType& t)
211 { m_retType=t; }
212 void SetName(const wxString& name)
213 { m_strName=name; }
f270e1dd
FM
214 void SetArgumentTypes(const wxArgumentTypeArray& arr)
215 { m_args=arr; }
5934cda1
FM
216 void SetConst(bool c = true)
217 { m_bConst=c; }
218 void SetStatic(bool c = true)
219 { m_bStatic=c; }
220 void SetVirtual(bool c = true)
221 { m_bVirtual=c; }
a7be99c8 222 void SetPureVirtual(bool c = true)
d5978709
FM
223 {
224 m_bPureVirtual=c;
225 if (c) m_bVirtual=c; // pure virtual => virtual
226 }
0403e5dc
FM
227 void SetDeprecated(bool c = true)
228 { m_bDeprecated=c; }
5934cda1
FM
229 void SetLocation(int lineNumber)
230 { m_nLine=lineNumber; }
03d4f7b9
FM
231 void SetAvailability(int nAvail)
232 { m_nAvailability=nAvail; }
8cd22478
FM
233 void SetAccessSpecifier(wxMethodAccessSpecifier spec)
234 { m_access=spec; }
5934cda1
FM
235
236public: // misc
237
238 bool operator==(const wxMethod&) const;
239 bool operator!=(const wxMethod& m) const
240 { return !(*this == m); }
241
97f0dbd6
FM
242 // this function works like operator== but tests everything:
243 // - method name
244 // - return type
245 // - argument types
246 // except for the method attributes (const,static,virtual,pureVirtual,deprecated)
247 bool MatchesExceptForAttributes(const wxMethod& m) const;
248
f3998820
FM
249 // returns true if this is a ctor which has default values for all its
250 // argument, thus is able to act also as default ctor
251 bool ActsAsDefaultCtor() const;
252
83fdf796 253 // dumps the contents of this class in the given stream
5934cda1
FM
254 void Dump(wxTextOutputStream& stream) const;
255
256protected:
257 wxType m_retType;
258 wxString m_strName;
f270e1dd 259 wxArgumentTypeArray m_args;
03d4f7b9
FM
260
261 // misc attributes:
5934cda1
FM
262 bool m_bConst;
263 bool m_bStatic;
264 bool m_bVirtual;
a7be99c8 265 bool m_bPureVirtual;
0403e5dc 266 bool m_bDeprecated;
03d4f7b9
FM
267
268 // m_nLine can be -1 if no location infos are available
5934cda1 269 int m_nLine;
03d4f7b9
FM
270
271 // this is a combination of wxPORT_* flags (see wxPortId) or wxPORT_UNKNOWN
272 // if this method should be available for all wxWidgets ports.
273 // NOTE: this is not used for comparing wxMethod objects
274 // (gccXML never gives this kind of info).
275 int m_nAvailability;
8cd22478
FM
276
277 // the access specifier for this method
278 wxMethodAccessSpecifier m_access;
5934cda1
FM
279};
280
281WX_DECLARE_OBJARRAY(wxMethod, wxMethodArray);
282WX_DEFINE_ARRAY(const wxMethod*, wxMethodPtrArray);
283
284
673ae68a
FM
285// we need wxClassPtrArray to be defined _before_ wxClass itself,
286// since wxClass uses wxClassPtrArray.
287class wxClass;
288WX_DEFINE_ARRAY(const wxClass*, wxClassPtrArray);
289
290class wxXmlInterface;
291
292
5934cda1
FM
293// ----------------------------------------------------------------------------
294// Represents a class of the wx API/interface.
295// ----------------------------------------------------------------------------
296class wxClass
297{
298public:
299 wxClass() {}
300 wxClass(const wxString& name, const wxString& headername)
301 : m_strName(name), m_strHeader(headername) {}
302
03d4f7b9
FM
303
304public: // setters
5934cda1
FM
305
306 void SetHeader(const wxString& header)
307 { m_strHeader=header; }
308 void SetName(const wxString& name)
309 { m_strName=name; }
03d4f7b9
FM
310 void SetAvailability(int nAvail)
311 { m_nAvailability=nAvail; }
673ae68a
FM
312 void SetParent(unsigned int k, const wxString& name)
313 { m_parents[k]=name; }
03d4f7b9
FM
314
315public: // getters
5934cda1
FM
316
317 bool IsOk() const
318 { return !m_strName.IsEmpty() && !m_methods.IsEmpty(); }
319
320 bool IsValidCtorForThisClass(const wxMethod& m) const;
321 bool IsValidDtorForThisClass(const wxMethod& m) const;
322
03d4f7b9
FM
323 wxString GetName() const
324 { return m_strName; }
325 wxString GetHeader() const
326 { return m_strHeader; }
327 wxString GetNameWithoutTemplate() const;
328
5934cda1
FM
329 unsigned int GetMethodCount() const
330 { return m_methods.GetCount(); }
331 wxMethod& GetMethod(unsigned int n) const
332 { return m_methods[n]; }
333 wxMethod& GetLastMethod() const
334 { return m_methods.Last(); }
335
03d4f7b9
FM
336 int GetAvailability() const
337 { return m_nAvailability; }
338
673ae68a
FM
339 //const wxClass *GetParent(unsigned int i) const
340 const wxString& GetParent(unsigned int i) const
341 { return m_parents[i]; }
342 unsigned int GetParentCount() const
343 { return m_parents.GetCount(); }
344
03d4f7b9
FM
345public: // misc
346
347 void AddMethod(const wxMethod& func)
348 { m_methods.Add(func); }
349
673ae68a
FM
350 void AddParent(const wxString& parent)//wxClass* parent)
351 { m_parents.Add(parent); }
352
5934cda1
FM
353 // returns a single result (the first, which is also the only
354 // one if CheckConsistency() return true)
355 const wxMethod* FindMethod(const wxMethod& m) const;
356
673ae68a
FM
357 // like FindMethod() but this one searches also recursively in
358 // the parents of this class.
359 const wxMethod* RecursiveUpwardFindMethod(const wxMethod& m,
360 const wxXmlInterface* allclasses) const;
361
5934cda1
FM
362 // returns an array of pointers to the overloaded methods with the
363 // same given name
673ae68a
FM
364 wxMethodPtrArray FindMethodsNamed(const wxString& name) const;
365
366 // like FindMethodsNamed() but this one searches also recursively in
367 // the parents of this class.
368 wxMethodPtrArray RecursiveUpwardFindMethodsNamed(const wxString& name,
369 const wxXmlInterface* allclasses) const;
5934cda1
FM
370
371 // dumps all methods to the given output stream
372 void Dump(wxTextOutputStream& stream) const;
373
374 // slow check
375 bool CheckConsistency() const;
376
377protected:
378 wxString m_strName;
379 wxString m_strHeader;
380 wxMethodArray m_methods;
03d4f7b9 381
673ae68a
FM
382 // name of the base classes: we store the names and not the pointers
383 // because this makes _much_ easier the parsing process!
384 // (basically because when parsing class X which derives from Y,
385 // we may have not parsed yet class Y!)
386 wxArrayString m_parents;
387
03d4f7b9
FM
388 // see the wxMethod::m_nAvailability field for more info
389 int m_nAvailability;
5934cda1
FM
390};
391
392WX_DECLARE_OBJARRAY(wxClass, wxClassArray);
5934cda1
FM
393
394
395
396// ----------------------------------------------------------------------------
397// wxXmlInterface
398// ----------------------------------------------------------------------------
399class wxXmlInterface
400{
401public:
402 wxXmlInterface() {}
403
404 const wxClass* FindClass(const wxString& classname) const
405 {
406 for (unsigned int i=0; i<m_classes.GetCount(); i++)
407 if (m_classes[i].GetName() == classname)
408 return &m_classes[i];
409 return NULL;
410 }
411
412 void Dump(const wxString& filename);
413
414 const wxClassArray& GetClasses() const
415 { return m_classes; }
416
417 unsigned int GetClassesCount() const
418 { return m_classes.GetCount(); }
419
420 unsigned int GetMethodCount() const
421 {
422 unsigned int methods = 0;
423 for (unsigned i=0; i < m_classes.GetCount(); i++)
424 methods += m_classes[i].GetMethodCount();
425 return methods;
426 }
427
7fbadf87
FM
428 // pass a full-path header filename:
429 wxClassPtrArray FindClassesDefinedIn(const wxString& headerfile) const;
430
5934cda1 431 void ShowProgress()
7f8fae98 432 { /*wxFprintf(stderr, ".");*/ }
5934cda1 433
7f8fae98
FM
434 // is this interface coherent?
435 bool CheckConsistency() const;
5934cda1
FM
436
437protected:
438 wxClassArray m_classes;
439};
440
441#if 1
74bda203
FM
442// for wxTypeIdHashMap, keys == gccxml IDs and values == associated type strings
443// e.g. key = "0x123f" and value = "const wxAboutDialogInfo&"
fdd4a897
FM
444WX_DECLARE_HASH_MAP( unsigned long, wxString,
445 wxIntegerHash, wxIntegerEqual,
446 wxTypeIdHashMap );
5570107a
FM
447
448WX_DECLARE_STRING_HASH_MAP( wxString, wxStringHashMap );
5934cda1
FM
449#else
450#include <map>
451typedef std::basic_string<char> stlString;
fdd4a897 452typedef std::map<unsigned long, stlString> wxTypeIdHashMap;
5934cda1
FM
453#endif
454
455
456// ----------------------------------------------------------------------------
457// Represents the real interface of wxWidgets
458// Loads it from the XML produced by gccXML: http://www.gccxml.org
459// ----------------------------------------------------------------------------
460class wxXmlGccInterface : public wxXmlInterface
461{
462public:
03d4f7b9
FM
463 wxXmlGccInterface()
464 {
465 // FIXME: we should retrieve this from the XML file!
466 // here we suppose the XML was created for the currently-running port
467 m_portId = wxPlatformInfo::Get().GetPortId();
468 }
5934cda1 469
5934cda1
FM
470 bool Parse(const wxString& filename);
471 bool ParseMethod(const wxXmlNode *p,
fdd4a897 472 const wxTypeIdHashMap& types,
5934cda1 473 wxMethod& m);
03d4f7b9
FM
474
475 wxPortId GetInterfacePort() const
476 { return m_portId; }
477
478 wxString GetInterfacePortName() const
479 { return wxPlatformInfo::GetPortIdName(m_portId, false); }
480
481protected:
482 // the port for which the gcc XML was generated
483 wxPortId m_portId;
5934cda1
FM
484};
485
486
487// ----------------------------------------------------------------------------
488// Represents the interface of the doxygen headers of wxWidgets
489// Loads it from the XML produced by Doxygen: http://www.doxygen.org
490// ----------------------------------------------------------------------------
491class wxXmlDoxygenInterface : public wxXmlInterface
492{
493public:
494 wxXmlDoxygenInterface() {}
495
496 // !!SPEEDUP-TODO!!
497 // Using wxXmlDocument::Load as is, the entire XML file is parsed
498 // and an entire tree of wxXmlNodes is built in memory.
499 // We need however only small portions of the Doxygen-generated XML: to speedup the
500 // processing we could detach the expat callbacks when we detect the beginning of a
501 // node we're not interested about, or just don't create a wxXmlNode for it!
502 // This needs a modification of wxXml API.
503
504 bool Parse(const wxString& filename);
505 bool ParseCompoundDefinition(const wxString& filename);
506 bool ParseMethod(const wxXmlNode*, wxMethod&, wxString& header);
5570107a
FM
507
508 // this class can take advantage of the preprocessor output to give
509 // a minor number of false positive warnings in the final comparison
510 void AddPreprocessorValue(const wxString& name, const wxString& val)
511 { m_preproc[name]=val; }
512
513protected:
514 wxStringHashMap m_preproc;
5934cda1
FM
515};
516
517
518
519#endif // _XMLPARSER_H_
520