4 * Copyright 2002 Lionel Ulmer
5 * Copyright 2005 Mike McCormack
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
23 #include "wine/port.h"
29 # include <libxml/parser.h>
30 # include <libxml/xmlerror.h>
31 # ifdef SONAME_LIBXSLT
32 # ifdef HAVE_LIBXSLT_PATTERN_H
33 # include <libxslt/pattern.h>
35 # ifdef HAVE_LIBXSLT_TRANSFORM_H
36 # include <libxslt/transform.h>
38 # include <libxslt/xsltutils.h>
39 # include <libxslt/variables.h>
40 # include <libxslt/xsltInternals.h>
52 #include "wine/unicode.h"
53 #include "wine/debug.h"
54 #include "wine/library.h"
56 #include "msxml_private.h"
58 WINE_DEFAULT_DEBUG_CHANNEL(msxml
);
60 HINSTANCE MSXML_hInstance
= NULL
;
64 void wineXmlCallbackLog(char const* caller
, xmlErrorLevel lvl
, char const* msg
, va_list ap
)
66 static const int max_size
= 200;
67 enum __wine_debug_class dbcl
;
74 dbcl
= __WINE_DBCL_TRACE
;
77 dbcl
= __WINE_DBCL_WARN
;
80 dbcl
= __WINE_DBCL_ERR
;
84 len
= vsnprintf(buff
, max_size
, msg
, ap
);
85 if (len
== -1 || len
>= max_size
) buff
[max_size
-1] = 0;
87 wine_dbg_log(dbcl
, &__wine_dbch_msxml
, caller
, "%s", buff
);
90 void wineXmlCallbackError(char const* caller
, xmlErrorPtr err
)
92 enum __wine_debug_class dbcl
;
96 case XML_ERR_NONE
: dbcl
= __WINE_DBCL_TRACE
; break;
97 case XML_ERR_WARNING
: dbcl
= __WINE_DBCL_WARN
; break;
98 default: dbcl
= __WINE_DBCL_ERR
; break;
101 wine_dbg_log(dbcl
, &__wine_dbch_msxml
, caller
, "error code %d", err
->code
);
103 wine_dbg_log(dbcl
, &__wine_dbch_msxml
, caller
, ": %s", err
->message
);
105 wine_dbg_log(dbcl
, &__wine_dbch_msxml
, caller
, "\n");
108 /* Support for loading xml files from a Wine Windows drive */
109 static int wineXmlMatchCallback (char const * filename
)
113 TRACE("%s\n", filename
);
116 * We will deal with loading XML files from the file system
117 * We only care about files that linux cannot find.
120 if(isalpha(filename
[0]) && filename
[1] == ':')
126 static void *wineXmlOpenCallback (char const * filename
)
128 BSTR sFilename
= bstr_from_xmlChar( (const xmlChar
*)filename
);
131 TRACE("%s\n", debugstr_w(sFilename
));
133 hFile
= CreateFileW(sFilename
, GENERIC_READ
,FILE_SHARE_READ
, NULL
,
134 OPEN_EXISTING
,FILE_ATTRIBUTE_NORMAL
, NULL
);
135 if(hFile
== INVALID_HANDLE_VALUE
) hFile
= 0;
136 SysFreeString(sFilename
);
140 static int wineXmlReadCallback(void * context
, char * buffer
, int len
)
144 TRACE("%p %s %d\n", context
, buffer
, len
);
146 if ((context
== NULL
) || (buffer
== NULL
))
149 if(!ReadFile( context
, buffer
,len
, &dwBytesRead
, NULL
))
151 ERR("Failed to read file\n");
155 TRACE("Read %d\n", dwBytesRead
);
160 static int wineXmlFileCloseCallback (void * context
)
162 return CloseHandle(context
) ? 0 : -1;
165 void* libxslt_handle
= NULL
;
166 #ifdef SONAME_LIBXSLT
167 # define DECL_FUNCPTR(f) typeof(f) * p##f = NULL
168 DECL_FUNCPTR(xsltApplyStylesheet
);
169 DECL_FUNCPTR(xsltApplyStylesheetUser
);
170 DECL_FUNCPTR(xsltCleanupGlobals
);
171 DECL_FUNCPTR(xsltFreeStylesheet
);
172 DECL_FUNCPTR(xsltFreeTransformContext
);
173 DECL_FUNCPTR(xsltNewTransformContext
);
174 DECL_FUNCPTR(xsltParseStylesheetDoc
);
175 DECL_FUNCPTR(xsltQuoteUserParams
);
176 DECL_FUNCPTR(xsltSaveResultTo
);
180 static void init_libxslt(void)
182 #ifdef SONAME_LIBXSLT
183 void (*pxsltInit
)(void); /* Missing in libxslt <= 1.1.14 */
185 libxslt_handle
= wine_dlopen(SONAME_LIBXSLT
, RTLD_NOW
, NULL
, 0);
189 #define LOAD_FUNCPTR(f, needed) \
190 if ((p##f = wine_dlsym(libxslt_handle, #f, NULL, 0)) == NULL) \
191 if (needed) { WARN("Can't find symbol %s\n", #f); goto sym_not_found; }
192 LOAD_FUNCPTR(xsltInit
, 0);
193 LOAD_FUNCPTR(xsltApplyStylesheet
, 1);
194 LOAD_FUNCPTR(xsltApplyStylesheetUser
, 1);
195 LOAD_FUNCPTR(xsltCleanupGlobals
, 1);
196 LOAD_FUNCPTR(xsltFreeStylesheet
, 1);
197 LOAD_FUNCPTR(xsltFreeTransformContext
, 1);
198 LOAD_FUNCPTR(xsltNewTransformContext
, 1);
199 LOAD_FUNCPTR(xsltParseStylesheetDoc
, 1);
200 LOAD_FUNCPTR(xsltQuoteUserParams
, 1);
201 LOAD_FUNCPTR(xsltSaveResultTo
, 1);
209 wine_dlclose(libxslt_handle
, NULL
, 0);
210 libxslt_handle
= NULL
;
214 #endif /* HAVE_LIBXML2 */
217 HRESULT WINAPI
DllCanUnloadNow(void)
223 BOOL WINAPI
DllMain(HINSTANCE hInstDLL
, DWORD fdwReason
, LPVOID reserved
)
225 MSXML_hInstance
= hInstDLL
;
229 case DLL_PROCESS_ATTACH
:
233 /* Set the default indent character to a single tab,
234 for this thread and as default for new threads */
235 xmlTreeIndentString
= "\t";
236 xmlThrDefTreeIndentString("\t");
238 /* Register callbacks for loading XML files */
239 if(xmlRegisterInputCallbacks(wineXmlMatchCallback
, wineXmlOpenCallback
,
240 wineXmlReadCallback
, wineXmlFileCloseCallback
) == -1)
241 WARN("Failed to register callbacks\n");
246 DisableThreadLibraryCalls(hInstDLL
);
248 case DLL_PROCESS_DETACH
:
251 #ifdef SONAME_LIBXSLT
254 pxsltCleanupGlobals();
255 wine_dlclose(libxslt_handle
, NULL
, 0);
258 /* Restore default Callbacks */
259 xmlCleanupInputCallbacks();
260 xmlRegisterDefaultInputCallbacks();
271 const char *debugstr_variant(const VARIANT
*v
)
282 return wine_dbg_sprintf("{VT_I1: %d}", V_I1(v
));
284 return wine_dbg_sprintf("{VT_I2: %d}", V_I2(v
));
286 return wine_dbg_sprintf("{VT_I4: %d}", V_I4(v
));
288 return wine_dbg_sprintf("{VT_INT: %d}", V_INT(v
));
290 return wine_dbg_sprintf("{VT_R8: %lf}", V_R8(v
));
292 return wine_dbg_sprintf("{VT_BSTR: %s}", debugstr_w(V_BSTR(v
)));
294 return wine_dbg_sprintf("{VT_DISPATCH: %p}", V_DISPATCH(v
));
296 return wine_dbg_sprintf("{VT_BOOL: %x}", V_BOOL(v
));
298 return wine_dbg_sprintf("{VT_UNKNOWN: %p}", V_UNKNOWN(v
));
300 return wine_dbg_sprintf("{VT_UINT: %u}", V_UINT(v
));
301 case VT_BSTR
|VT_BYREF
:
302 return wine_dbg_sprintf("{VT_BSTR|VT_BYREF: ptr %p, data %s}",
303 V_BSTRREF(v
), debugstr_w(V_BSTRREF(v
) ? *V_BSTRREF(v
) : NULL
));
305 return wine_dbg_sprintf("{VT_ERROR: 0x%08x}", V_ERROR(v
));
306 case VT_VARIANT
|VT_BYREF
:
307 return wine_dbg_sprintf("{VT_VARIANT|VT_BYREF: %s}", debugstr_variant(V_VARIANTREF(v
)));
308 case VT_UI1
|VT_ARRAY
:
309 return "{VT_UI1|VT_ARRAY}";
311 return wine_dbg_sprintf("{vt %d}", V_VT(v
));
315 /***********************************************************************
316 * DllRegisterServer (MSXML3.@)
318 HRESULT WINAPI
DllRegisterServer(void)
320 return __wine_register_resources( MSXML_hInstance
);
323 /***********************************************************************
324 * DllUnregisterServer (MSXML3.@)
326 HRESULT WINAPI
DllUnregisterServer(void)
328 return __wine_unregister_resources( MSXML_hInstance
);