307 lines
7.6 KiB
C
307 lines
7.6 KiB
C
/*
|
|
* MSXML Class Factory
|
|
*
|
|
* Copyright 2002 Lionel Ulmer
|
|
* Copyright 2005 Mike McCormack
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "wine/port.h"
|
|
|
|
#define COBJMACROS
|
|
|
|
#include <stdarg.h>
|
|
#ifdef HAVE_LIBXML2
|
|
# include <libxml/parser.h>
|
|
# include <libxml/xmlerror.h>
|
|
# ifdef SONAME_LIBXSLT
|
|
# ifdef HAVE_LIBXSLT_PATTERN_H
|
|
# include <libxslt/pattern.h>
|
|
# endif
|
|
# ifdef HAVE_LIBXSLT_TRANSFORM_H
|
|
# include <libxslt/transform.h>
|
|
# endif
|
|
# include <libxslt/xsltutils.h>
|
|
# include <libxslt/xsltInternals.h>
|
|
# endif
|
|
#endif
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winuser.h"
|
|
#include "ole2.h"
|
|
#include "rpcproxy.h"
|
|
#include "msxml.h"
|
|
#include "msxml6.h"
|
|
|
|
#include "wine/unicode.h"
|
|
#include "wine/debug.h"
|
|
#include "wine/library.h"
|
|
|
|
#include "msxml_private.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(msxml);
|
|
|
|
HINSTANCE MSXML_hInstance = NULL;
|
|
|
|
#ifdef HAVE_LIBXML2
|
|
|
|
void wineXmlCallbackLog(char const* caller, xmlErrorLevel lvl, char const* msg, va_list ap)
|
|
{
|
|
char* buf = NULL;
|
|
int len = 32, needed;
|
|
enum __wine_debug_class dbcl = __WINE_DBCL_ERR;
|
|
switch (lvl)
|
|
{
|
|
case XML_ERR_NONE:
|
|
dbcl = __WINE_DBCL_TRACE;
|
|
break;
|
|
case XML_ERR_WARNING:
|
|
dbcl = __WINE_DBCL_WARN;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
do
|
|
{
|
|
heap_free(buf);
|
|
buf = heap_alloc(len);
|
|
needed = vsnprintf(buf, len, msg, ap);
|
|
if (needed == -1)
|
|
len *= 2;
|
|
else if (needed >= len)
|
|
len = needed + 1;
|
|
else
|
|
needed = 0;
|
|
}
|
|
while (needed);
|
|
|
|
wine_dbg_log(dbcl, &__wine_dbch_msxml, caller, "%s", buf);
|
|
heap_free(buf);
|
|
}
|
|
|
|
void wineXmlCallbackError(char const* caller, xmlErrorPtr err)
|
|
{
|
|
enum __wine_debug_class dbcl;
|
|
|
|
switch (err->level)
|
|
{
|
|
case XML_ERR_NONE: dbcl = __WINE_DBCL_TRACE; break;
|
|
case XML_ERR_WARNING: dbcl = __WINE_DBCL_WARN; break;
|
|
default: dbcl = __WINE_DBCL_ERR; break;
|
|
}
|
|
wine_dbg_log(dbcl, &__wine_dbch_msxml, caller, "%s", err->message);
|
|
}
|
|
|
|
/* Support for loading xml files from a Wine Windows drive */
|
|
static int wineXmlMatchCallback (char const * filename)
|
|
{
|
|
int nRet = 0;
|
|
|
|
TRACE("%s\n", filename);
|
|
|
|
/*
|
|
* We will deal with loading XML files from the file system
|
|
* We only care about files that linux cannot find.
|
|
* e.g. C:,D: etc
|
|
*/
|
|
if(isalpha(filename[0]) && filename[1] == ':')
|
|
nRet = 1;
|
|
|
|
return nRet;
|
|
}
|
|
|
|
static void *wineXmlOpenCallback (char const * filename)
|
|
{
|
|
BSTR sFilename = bstr_from_xmlChar( (const xmlChar*)filename);
|
|
HANDLE hFile;
|
|
|
|
TRACE("%s\n", debugstr_w(sFilename));
|
|
|
|
hFile = CreateFileW(sFilename, GENERIC_READ,FILE_SHARE_READ, NULL,
|
|
OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL, NULL);
|
|
if(hFile == INVALID_HANDLE_VALUE) hFile = 0;
|
|
SysFreeString(sFilename);
|
|
return hFile;
|
|
}
|
|
|
|
static int wineXmlReadCallback(void * context, char * buffer, int len)
|
|
{
|
|
DWORD dwBytesRead;
|
|
|
|
TRACE("%p %s %d\n", context, buffer, len);
|
|
|
|
if ((context == NULL) || (buffer == NULL))
|
|
return(-1);
|
|
|
|
if(!ReadFile( context, buffer,len, &dwBytesRead, NULL))
|
|
{
|
|
ERR("Failed to read file\n");
|
|
return -1;
|
|
}
|
|
|
|
TRACE("Read %d\n", dwBytesRead);
|
|
|
|
return dwBytesRead;
|
|
}
|
|
|
|
static int wineXmlFileCloseCallback (void * context)
|
|
{
|
|
return CloseHandle(context) ? 0 : -1;
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
HRESULT WINAPI DllCanUnloadNow(void)
|
|
{
|
|
return S_FALSE;
|
|
}
|
|
|
|
|
|
void* libxslt_handle = NULL;
|
|
#ifdef SONAME_LIBXSLT
|
|
# define DECL_FUNCPTR(f) typeof(f) * p##f = NULL
|
|
DECL_FUNCPTR(xsltApplyStylesheet);
|
|
DECL_FUNCPTR(xsltCleanupGlobals);
|
|
DECL_FUNCPTR(xsltFreeStylesheet);
|
|
DECL_FUNCPTR(xsltParseStylesheetDoc);
|
|
# undef DECL_FUNCPTR
|
|
#endif
|
|
|
|
static void init_libxslt(void)
|
|
{
|
|
#ifdef SONAME_LIBXSLT
|
|
void (*pxsltInit)(void); /* Missing in libxslt <= 1.1.14 */
|
|
|
|
libxslt_handle = wine_dlopen(SONAME_LIBXSLT, RTLD_NOW, NULL, 0);
|
|
if (!libxslt_handle)
|
|
return;
|
|
|
|
#define LOAD_FUNCPTR(f, needed) if ((p##f = wine_dlsym(libxslt_handle, #f, NULL, 0)) == NULL && needed) { WARN("Can't find symbol %s\n", #f); goto sym_not_found; }
|
|
LOAD_FUNCPTR(xsltInit, 0);
|
|
LOAD_FUNCPTR(xsltApplyStylesheet, 1);
|
|
LOAD_FUNCPTR(xsltCleanupGlobals, 1);
|
|
LOAD_FUNCPTR(xsltFreeStylesheet, 1);
|
|
LOAD_FUNCPTR(xsltParseStylesheetDoc, 1);
|
|
#undef LOAD_FUNCPTR
|
|
|
|
if (pxsltInit)
|
|
pxsltInit();
|
|
return;
|
|
|
|
sym_not_found:
|
|
wine_dlclose(libxslt_handle, NULL, 0);
|
|
libxslt_handle = NULL;
|
|
#endif
|
|
}
|
|
|
|
|
|
BOOL WINAPI DllMain(HINSTANCE hInstDLL, DWORD fdwReason, LPVOID lpv)
|
|
{
|
|
MSXML_hInstance = hInstDLL;
|
|
|
|
switch(fdwReason)
|
|
{
|
|
case DLL_PROCESS_ATTACH:
|
|
#ifdef HAVE_LIBXML2
|
|
xmlInitParser();
|
|
|
|
/* Set the default indent character to a single tab,
|
|
for this thread and as default for new threads */
|
|
xmlTreeIndentString = "\t";
|
|
xmlThrDefTreeIndentString("\t");
|
|
|
|
/* Register callbacks for loading XML files */
|
|
if(xmlRegisterInputCallbacks(wineXmlMatchCallback, wineXmlOpenCallback,
|
|
wineXmlReadCallback, wineXmlFileCloseCallback) == -1)
|
|
WARN("Failed to register callbacks\n");
|
|
|
|
schemasInit();
|
|
#endif
|
|
init_libxslt();
|
|
DisableThreadLibraryCalls(hInstDLL);
|
|
break;
|
|
case DLL_PROCESS_DETACH:
|
|
#ifdef SONAME_LIBXSLT
|
|
if (libxslt_handle)
|
|
{
|
|
pxsltCleanupGlobals();
|
|
wine_dlclose(libxslt_handle, NULL, 0);
|
|
libxslt_handle = NULL;
|
|
}
|
|
#endif
|
|
#ifdef HAVE_LIBXML2
|
|
/* Restore default Callbacks */
|
|
xmlCleanupInputCallbacks();
|
|
xmlRegisterDefaultInputCallbacks();
|
|
|
|
xmlCleanupParser();
|
|
schemasCleanup();
|
|
#endif
|
|
release_typelib();
|
|
break;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
const char *debugstr_variant(const VARIANT *v)
|
|
{
|
|
if(!v)
|
|
return "(null)";
|
|
|
|
switch(V_VT(v)) {
|
|
case VT_EMPTY:
|
|
return "{VT_EMPTY}";
|
|
case VT_NULL:
|
|
return "{VT_NULL}";
|
|
case VT_I4:
|
|
return wine_dbg_sprintf("{VT_I4: %d}", V_I4(v));
|
|
case VT_R8:
|
|
return wine_dbg_sprintf("{VT_R8: %lf}", V_R8(v));
|
|
case VT_BSTR:
|
|
return wine_dbg_sprintf("{VT_BSTR: %s}", debugstr_w(V_BSTR(v)));
|
|
case VT_DISPATCH:
|
|
return wine_dbg_sprintf("{VT_DISPATCH: %p}", V_DISPATCH(v));
|
|
case VT_BOOL:
|
|
return wine_dbg_sprintf("{VT_BOOL: %x}", V_BOOL(v));
|
|
case VT_UNKNOWN:
|
|
return wine_dbg_sprintf("{VT_UNKNOWN: %p}", V_UNKNOWN(v));
|
|
case VT_UINT:
|
|
return wine_dbg_sprintf("{VT_UINT: %u}", V_UINT(v));
|
|
default:
|
|
return wine_dbg_sprintf("{vt %d}", V_VT(v));
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllRegisterServer (MSXML3.@)
|
|
*/
|
|
HRESULT WINAPI DllRegisterServer(void)
|
|
{
|
|
return __wine_register_resources( MSXML_hInstance, NULL );
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllUnregisterServer (MSXML3.@)
|
|
*/
|
|
HRESULT WINAPI DllUnregisterServer(void)
|
|
{
|
|
return __wine_unregister_resources( MSXML_hInstance, NULL );
|
|
}
|