#include "wx/xml/xmlio.h"
#include <libxml/parser.h>
+#include <libxml/SAX.h>
+
+
+// wxWindows SAX handlers for bugs reporting:
+
+static void wxXmlParserError(void *ctx, const char *msg, ...)
+{
+ wxString text;
+ xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
+ if (ctxt->input)
+ text.Printf( _("XML parser error at line %d: "), ctxt->input->line );
+ va_list args;
+ wxString tmp;
+ va_start(args, msg);
+ tmp.PrintfV( msg, args );
+ va_end(args);
+ text += tmp;
+ wxLogError( text.c_str() );
+}
+
+static void wxXmlParserWarning(void *ctx, const char *msg, ...)
+{
+ wxString text;
+ xmlParserCtxtPtr ctxt = (xmlParserCtxtPtr) ctx;
+ if (ctxt->input)
+ text.Printf( _("XML parser warning at line %d: "), ctxt->input->line );
+ va_list args;
+ wxString tmp;
+ va_start(args, msg);
+ tmp.PrintfV( msg, args );
+ va_end(args);
+ text += tmp;
+ wxLogWarning( text.c_str() );
+}
+
+static xmlSAXHandler gs_wxXmlSAXHandler;
+
+
// dynamically loaded functions from libxml:
typedef xmlParserCtxtPtr (*type_xmlCreatePushParserCtxt)
typedef xmlNodePtr (*type_xmlDocSetRootElement)(xmlDocPtr doc, xmlNodePtr root);
typedef void (*(*type_xmlFree))(void *);
typedef int (*type_xmlKeepBlanksDefault)(int);
+typedef void (*type_xmlInitParser)(void);
+typedef void (*type_xmlCleanupParser)(void);
+typedef xmlSAXHandler *type_xmlDefaultSAXHandler;
static struct
{
type_xmlDocSetRootElement xmlDocSetRootElement;
type_xmlFree xmlFree;
type_xmlKeepBlanksDefault xmlKeepBlanksDefault;
+ type_xmlInitParser xmlInitParser;
+ type_xmlCleanupParser xmlCleanupParser;
+ type_xmlDefaultSAXHandler xmlDefaultSAXHandler;
} gs_libxmlDLL;
static bool gs_libxmlLoaded = FALSE;
{
if (gs_libxmlLoaded)
{
+ // Check for CleanupParser ..may have failed before initialised
+ // during LOAD_SYMBOL in LoadLibxml()
+ if (gs_libxmlDLL.xmlCleanupParser)
+ gs_libxmlDLL.xmlCleanupParser();
wxDllLoader::UnloadLibrary(gs_libxmlDLL.Handle);
}
gs_libxmlLoaded = FALSE;
wxLogNull lg;
#ifdef __UNIX__
gs_libxmlDLL.Handle =
- wxDllLoader::LoadLibrary(_T("wxlibxml.so.2"), &gs_libxmlLoaded);
+ wxDllLoader::LoadLibrary(wxT("wxlibxml.so.2"), &gs_libxmlLoaded);
if (!gs_libxmlLoaded) gs_libxmlDLL.Handle =
- wxDllLoader::LoadLibrary(_T("libxml.so.2"), &gs_libxmlLoaded);
+ wxDllLoader::LoadLibrary(wxT("libxml.so.2"), &gs_libxmlLoaded);
#endif
#ifdef __WXMSW__
gs_libxmlDLL.Handle =
- wxDllLoader::LoadLibrary(_T("wxlibxml2.dll"), &gs_libxmlLoaded);
+ wxDllLoader::LoadLibrary(wxT("wxlibxml2.dll"), &gs_libxmlLoaded);
if (!gs_libxmlLoaded) gs_libxmlDLL.Handle =
- wxDllLoader::LoadLibrary(_T("libxml2.dll"), &gs_libxmlLoaded);
+ wxDllLoader::LoadLibrary(wxT("libxml2.dll"), &gs_libxmlLoaded);
#endif
}
#define LOAD_SYMBOL(sym) \
gs_libxmlDLL.sym = \
- (type_##sym)wxDllLoader::GetSymbol(gs_libxmlDLL.Handle, _T(#sym)); \
+ (type_##sym)wxDllLoader::GetSymbol(gs_libxmlDLL.Handle, wxT(#sym)); \
if (!gs_libxmlDLL.sym) \
{ \
ReleaseLibxml(); \
LOAD_SYMBOL(xmlDocSetRootElement)
LOAD_SYMBOL(xmlFree)
LOAD_SYMBOL(xmlKeepBlanksDefault)
+ LOAD_SYMBOL(xmlInitParser)
+ LOAD_SYMBOL(xmlCleanupParser)
+ LOAD_SYMBOL(xmlDefaultSAXHandler)
#undef LOAD_SYMBOL
gs_libxmlLoadFailed = FALSE;
+
+ gs_libxmlDLL.xmlInitParser();
+ memcpy(&gs_wxXmlSAXHandler, gs_libxmlDLL.xmlDefaultSAXHandler,
+ sizeof(xmlSAXHandler));
+ gs_wxXmlSAXHandler.error = wxXmlParserError;
+ gs_wxXmlSAXHandler.fatalError = wxXmlParserError;
+ gs_wxXmlSAXHandler.warning = wxXmlParserWarning;
return TRUE;
}
{
bool okay = TRUE;
gs_libxmlDLL.xmlKeepBlanksDefault(0);
- ctxt = gs_libxmlDLL.xmlCreatePushParserCtxt(NULL, NULL,
- buffer, res, ""/*docname*/);
+ ctxt = gs_libxmlDLL.xmlCreatePushParserCtxt(&gs_wxXmlSAXHandler,
+ NULL, buffer, res, ""/*docname*/);
while ((res = stream.Read(buffer, 1024).LastRead()) > 0)
if (gs_libxmlDLL.xmlParseChunk(ctxt, buffer, res, 0) != 0)
okay = FALSE;