2 * Copyright 2011 Jacek Caban for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
25 #include "wine/debug.h"
27 WINE_DEFAULT_DEBUG_CHANNEL(vbscript);
31 #define CTXARG_T DWORDLONG
32 #define IActiveScriptParseVtbl IActiveScriptParse64Vtbl
36 #define CTXARG_T DWORD
37 #define IActiveScriptParseVtbl IActiveScriptParse32Vtbl
42 IActiveScript IActiveScript_iface;
43 IActiveScriptParse IActiveScriptParse_iface;
44 IObjectSafety IObjectSafety_iface;
50 IActiveScriptSite *site;
56 static void change_state(VBScript *This, SCRIPTSTATE state)
58 if(This->state == state)
63 IActiveScriptSite_OnStateChange(This->site, state);
66 static void exec_queued_code(VBScript *This)
71 static HRESULT set_ctx_site(VBScript *This)
75 This->ctx->lcid = This->lcid;
77 hres = init_global(This->ctx);
81 IActiveScriptSite_AddRef(This->site);
82 This->ctx->site = This->site;
84 change_state(This, SCRIPTSTATE_INITIALIZED);
88 static void destroy_script(script_ctx_t *ctx)
91 IActiveScriptSite_Release(ctx->site);
93 IDispatchEx_Release(&ctx->script_obj->IDispatchEx_iface);
97 static void decrease_state(VBScript *This, SCRIPTSTATE state)
100 case SCRIPTSTATE_CONNECTED:
101 change_state(This, SCRIPTSTATE_DISCONNECTED);
102 if(state == SCRIPTSTATE_DISCONNECTED)
105 case SCRIPTSTATE_STARTED:
106 case SCRIPTSTATE_DISCONNECTED:
107 if(This->state == SCRIPTSTATE_DISCONNECTED)
108 change_state(This, SCRIPTSTATE_INITIALIZED);
109 if(state == SCRIPTSTATE_INITIALIZED)
112 case SCRIPTSTATE_INITIALIZED:
113 case SCRIPTSTATE_UNINITIALIZED:
114 change_state(This, state);
117 IActiveScriptSite_Release(This->site);
123 if(state == SCRIPTSTATE_CLOSED) {
124 destroy_script(This->ctx);
134 static inline VBScript *impl_from_IActiveScript(IActiveScript *iface)
136 return CONTAINING_RECORD(iface, VBScript, IActiveScript_iface);
139 static HRESULT WINAPI VBScript_QueryInterface(IActiveScript *iface, REFIID riid, void **ppv)
141 VBScript *This = impl_from_IActiveScript(iface);
143 if(IsEqualGUID(riid, &IID_IUnknown)) {
144 TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
145 *ppv = &This->IActiveScript_iface;
146 }else if(IsEqualGUID(riid, &IID_IActiveScript)) {
147 TRACE("(%p)->(IID_IActiveScript %p)\n", This, ppv);
148 *ppv = &This->IActiveScript_iface;
149 }else if(IsEqualGUID(riid, &IID_IActiveScriptParse)) {
150 TRACE("(%p)->(IID_IActiveScriptParse %p)\n", This, ppv);
151 *ppv = &This->IActiveScriptParse_iface;
152 }else if(IsEqualGUID(riid, &IID_IObjectSafety)) {
153 TRACE("(%p)->(IID_IObjectSafety %p)\n", This, ppv);
154 *ppv = &This->IObjectSafety_iface;
156 FIXME("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
158 return E_NOINTERFACE;
161 IUnknown_AddRef((IUnknown*)*ppv);
165 static ULONG WINAPI VBScript_AddRef(IActiveScript *iface)
167 VBScript *This = impl_from_IActiveScript(iface);
168 LONG ref = InterlockedIncrement(&This->ref);
170 TRACE("(%p) ref=%d\n", This, ref);
175 static ULONG WINAPI VBScript_Release(IActiveScript *iface)
177 VBScript *This = impl_from_IActiveScript(iface);
178 LONG ref = InterlockedDecrement(&This->ref);
180 TRACE("(%p) ref=%d\n", iface, ref);
184 IActiveScriptSite_Release(This->site);
191 static HRESULT WINAPI VBScript_SetScriptSite(IActiveScript *iface, IActiveScriptSite *pass)
193 VBScript *This = impl_from_IActiveScript(iface);
197 TRACE("(%p)->(%p)\n", This, pass);
205 if(InterlockedCompareExchange(&This->thread_id, GetCurrentThreadId(), 0))
209 IActiveScriptSite_AddRef(This->site);
211 hres = IActiveScriptSite_GetLCID(This->site, &lcid);
215 return This->ctx ? set_ctx_site(This) : S_OK;
218 static HRESULT WINAPI VBScript_GetScriptSite(IActiveScript *iface, REFIID riid,
221 VBScript *This = impl_from_IActiveScript(iface);
222 FIXME("(%p)->()\n", This);
226 static HRESULT WINAPI VBScript_SetScriptState(IActiveScript *iface, SCRIPTSTATE ss)
228 VBScript *This = impl_from_IActiveScript(iface);
230 TRACE("(%p)->(%d)\n", This, ss);
232 if(This->thread_id && GetCurrentThreadId() != This->thread_id)
235 if(ss == SCRIPTSTATE_UNINITIALIZED) {
236 if(This->state == SCRIPTSTATE_CLOSED)
239 decrease_state(This, SCRIPTSTATE_UNINITIALIZED);
247 case SCRIPTSTATE_STARTED:
248 case SCRIPTSTATE_CONNECTED: /* FIXME */
249 if(This->state == SCRIPTSTATE_CLOSED)
252 exec_queued_code(This);
254 case SCRIPTSTATE_INITIALIZED:
255 FIXME("unimplemented SCRIPTSTATE_INITIALIZED\n");
258 FIXME("unimplemented state %d\n", ss);
262 change_state(This, ss);
266 static HRESULT WINAPI VBScript_GetScriptState(IActiveScript *iface, SCRIPTSTATE *pssState)
268 VBScript *This = impl_from_IActiveScript(iface);
270 TRACE("(%p)->(%p)\n", This, pssState);
275 if(This->thread_id && This->thread_id != GetCurrentThreadId())
278 *pssState = This->state;
282 static HRESULT WINAPI VBScript_Close(IActiveScript *iface)
284 VBScript *This = impl_from_IActiveScript(iface);
286 TRACE("(%p)->()\n", This);
288 if(This->thread_id && This->thread_id != GetCurrentThreadId())
291 decrease_state(This, SCRIPTSTATE_CLOSED);
295 static HRESULT WINAPI VBScript_AddNamedItem(IActiveScript *iface, LPCOLESTR pstrName, DWORD dwFlags)
297 VBScript *This = impl_from_IActiveScript(iface);
298 FIXME("(%p)->(%s %x)\n", This, debugstr_w(pstrName), dwFlags);
302 static HRESULT WINAPI VBScript_AddTypeLib(IActiveScript *iface, REFGUID rguidTypeLib,
303 DWORD dwMajor, DWORD dwMinor, DWORD dwFlags)
305 VBScript *This = impl_from_IActiveScript(iface);
306 FIXME("(%p)->()\n", This);
310 static HRESULT WINAPI VBScript_GetScriptDispatch(IActiveScript *iface, LPCOLESTR pstrItemName, IDispatch **ppdisp)
312 VBScript *This = impl_from_IActiveScript(iface);
314 TRACE("(%p)->(%p)\n", This, ppdisp);
319 if(This->thread_id != GetCurrentThreadId() || !This->ctx->script_obj) {
324 *ppdisp = (IDispatch*)&This->ctx->script_obj->IDispatchEx_iface;
325 IDispatch_AddRef(*ppdisp);
329 static HRESULT WINAPI VBScript_GetCurrentScriptThreadID(IActiveScript *iface,
330 SCRIPTTHREADID *pstridThread)
332 VBScript *This = impl_from_IActiveScript(iface);
333 FIXME("(%p)->()\n", This);
337 static HRESULT WINAPI VBScript_GetScriptThreadID(IActiveScript *iface,
338 DWORD dwWin32ThreadId, SCRIPTTHREADID *pstidThread)
340 VBScript *This = impl_from_IActiveScript(iface);
341 FIXME("(%p)->()\n", This);
345 static HRESULT WINAPI VBScript_GetScriptThreadState(IActiveScript *iface,
346 SCRIPTTHREADID stidThread, SCRIPTTHREADSTATE *pstsState)
348 VBScript *This = impl_from_IActiveScript(iface);
349 FIXME("(%p)->()\n", This);
353 static HRESULT WINAPI VBScript_InterruptScriptThread(IActiveScript *iface,
354 SCRIPTTHREADID stidThread, const EXCEPINFO *pexcepinfo, DWORD dwFlags)
356 VBScript *This = impl_from_IActiveScript(iface);
357 FIXME("(%p)->()\n", This);
361 static HRESULT WINAPI VBScript_Clone(IActiveScript *iface, IActiveScript **ppscript)
363 VBScript *This = impl_from_IActiveScript(iface);
364 FIXME("(%p)->()\n", This);
368 static const IActiveScriptVtbl VBScriptVtbl = {
369 VBScript_QueryInterface,
372 VBScript_SetScriptSite,
373 VBScript_GetScriptSite,
374 VBScript_SetScriptState,
375 VBScript_GetScriptState,
377 VBScript_AddNamedItem,
379 VBScript_GetScriptDispatch,
380 VBScript_GetCurrentScriptThreadID,
381 VBScript_GetScriptThreadID,
382 VBScript_GetScriptThreadState,
383 VBScript_InterruptScriptThread,
387 static inline VBScript *impl_from_IActiveScriptParse(IActiveScriptParse *iface)
389 return CONTAINING_RECORD(iface, VBScript, IActiveScriptParse_iface);
392 static HRESULT WINAPI VBScriptParse_QueryInterface(IActiveScriptParse *iface, REFIID riid, void **ppv)
394 VBScript *This = impl_from_IActiveScriptParse(iface);
395 return IActiveScript_QueryInterface(&This->IActiveScript_iface, riid, ppv);
398 static ULONG WINAPI VBScriptParse_AddRef(IActiveScriptParse *iface)
400 VBScript *This = impl_from_IActiveScriptParse(iface);
401 return IActiveScript_AddRef(&This->IActiveScript_iface);
404 static ULONG WINAPI VBScriptParse_Release(IActiveScriptParse *iface)
406 VBScript *This = impl_from_IActiveScriptParse(iface);
407 return IActiveScript_Release(&This->IActiveScript_iface);
410 static HRESULT WINAPI VBScriptParse_InitNew(IActiveScriptParse *iface)
412 VBScript *This = impl_from_IActiveScriptParse(iface);
413 script_ctx_t *ctx, *old_ctx;
415 TRACE("(%p)\n", This);
420 ctx = heap_alloc_zero(sizeof(script_ctx_t));
422 return E_OUTOFMEMORY;
424 old_ctx = InterlockedCompareExchangePointer((void**)&This->ctx, ctx, NULL);
430 return This->site ? set_ctx_site(This) : S_OK;
433 static HRESULT WINAPI VBScriptParse_AddScriptlet(IActiveScriptParse *iface,
434 LPCOLESTR pstrDefaultName, LPCOLESTR pstrCode, LPCOLESTR pstrItemName,
435 LPCOLESTR pstrSubItemName, LPCOLESTR pstrEventName, LPCOLESTR pstrDelimiter,
436 CTXARG_T dwSourceContextCookie, ULONG ulStartingLineNumber, DWORD dwFlags,
437 BSTR *pbstrName, EXCEPINFO *pexcepinfo)
439 VBScript *This = impl_from_IActiveScriptParse(iface);
440 FIXME("(%p)->(%s %s %s %s %s %s %s %u %x %p %p)\n", This, debugstr_w(pstrDefaultName),
441 debugstr_w(pstrCode), debugstr_w(pstrItemName), debugstr_w(pstrSubItemName),
442 debugstr_w(pstrEventName), debugstr_w(pstrDelimiter), wine_dbgstr_longlong(dwSourceContextCookie),
443 ulStartingLineNumber, dwFlags, pbstrName, pexcepinfo);
447 static HRESULT WINAPI VBScriptParse_ParseScriptText(IActiveScriptParse *iface,
448 LPCOLESTR pstrCode, LPCOLESTR pstrItemName, IUnknown *punkContext,
449 LPCOLESTR pstrDelimiter, CTXARG_T dwSourceContextCookie, ULONG ulStartingLine,
450 DWORD dwFlags, VARIANT *pvarResult, EXCEPINFO *pexcepinfo)
452 VBScript *This = impl_from_IActiveScriptParse(iface);
453 FIXME("(%p)->(%s %s %p %s %s %u %x %p %p)\n", This, debugstr_w(pstrCode),
454 debugstr_w(pstrItemName), punkContext, debugstr_w(pstrDelimiter),
455 wine_dbgstr_longlong(dwSourceContextCookie), ulStartingLine, dwFlags, pvarResult, pexcepinfo);
459 static const IActiveScriptParseVtbl VBScriptParseVtbl = {
460 VBScriptParse_QueryInterface,
461 VBScriptParse_AddRef,
462 VBScriptParse_Release,
463 VBScriptParse_InitNew,
464 VBScriptParse_AddScriptlet,
465 VBScriptParse_ParseScriptText
468 static inline VBScript *impl_from_IObjectSafety(IObjectSafety *iface)
470 return CONTAINING_RECORD(iface, VBScript, IObjectSafety_iface);
473 static HRESULT WINAPI VBScriptSafety_QueryInterface(IObjectSafety *iface, REFIID riid, void **ppv)
475 VBScript *This = impl_from_IObjectSafety(iface);
476 return IActiveScript_QueryInterface(&This->IActiveScript_iface, riid, ppv);
479 static ULONG WINAPI VBScriptSafety_AddRef(IObjectSafety *iface)
481 VBScript *This = impl_from_IObjectSafety(iface);
482 return IActiveScript_AddRef(&This->IActiveScript_iface);
485 static ULONG WINAPI VBScriptSafety_Release(IObjectSafety *iface)
487 VBScript *This = impl_from_IObjectSafety(iface);
488 return IActiveScript_Release(&This->IActiveScript_iface);
491 #define SUPPORTED_OPTIONS (INTERFACESAFE_FOR_UNTRUSTED_DATA|INTERFACE_USES_DISPEX|INTERFACE_USES_SECURITY_MANAGER)
493 static HRESULT WINAPI VBScriptSafety_GetInterfaceSafetyOptions(IObjectSafety *iface, REFIID riid,
494 DWORD *pdwSupportedOptions, DWORD *pdwEnabledOptions)
496 VBScript *This = impl_from_IObjectSafety(iface);
498 TRACE("(%p)->(%s %p %p)\n", This, debugstr_guid(riid), pdwSupportedOptions, pdwEnabledOptions);
500 if(!pdwSupportedOptions || !pdwEnabledOptions)
503 *pdwSupportedOptions = SUPPORTED_OPTIONS;
504 *pdwEnabledOptions = This->safeopt;
508 static HRESULT WINAPI VBScriptSafety_SetInterfaceSafetyOptions(IObjectSafety *iface, REFIID riid,
509 DWORD dwOptionSetMask, DWORD dwEnabledOptions)
511 VBScript *This = impl_from_IObjectSafety(iface);
513 TRACE("(%p)->(%s %x %x)\n", This, debugstr_guid(riid), dwOptionSetMask, dwEnabledOptions);
515 if(dwOptionSetMask & ~SUPPORTED_OPTIONS)
518 This->safeopt = (dwEnabledOptions & dwOptionSetMask) | (This->safeopt & ~dwOptionSetMask) | INTERFACE_USES_DISPEX;
522 static const IObjectSafetyVtbl VBScriptSafetyVtbl = {
523 VBScriptSafety_QueryInterface,
524 VBScriptSafety_AddRef,
525 VBScriptSafety_Release,
526 VBScriptSafety_GetInterfaceSafetyOptions,
527 VBScriptSafety_SetInterfaceSafetyOptions
530 HRESULT WINAPI VBScriptFactory_CreateInstance(IClassFactory *iface, IUnknown *pUnkOuter, REFIID riid, void **ppv)
535 TRACE("(%p %s %p)\n", pUnkOuter, debugstr_guid(riid), ppv);
537 ret = heap_alloc_zero(sizeof(*ret));
539 return E_OUTOFMEMORY;
541 ret->IActiveScript_iface.lpVtbl = &VBScriptVtbl;
542 ret->IActiveScriptParse_iface.lpVtbl = &VBScriptParseVtbl;
543 ret->IObjectSafety_iface.lpVtbl = &VBScriptSafetyVtbl;
546 ret->state = SCRIPTSTATE_UNINITIALIZED;
547 ret->safeopt = INTERFACE_USES_DISPEX;
549 hres = IActiveScript_QueryInterface(&ret->IActiveScript_iface, riid, ppv);
550 IActiveScript_Release(&ret->IActiveScript_iface);