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 inline BOOL is_started(VBScript *This)
68 return This->state == SCRIPTSTATE_STARTED
69 || This->state == SCRIPTSTATE_CONNECTED
70 || This->state == SCRIPTSTATE_DISCONNECTED;
73 static HRESULT exec_global_code(script_ctx_t *ctx, vbscode_t *code)
77 code->global_executed = TRUE;
79 IActiveScriptSite_OnEnterScript(ctx->site);
80 hres = exec_script(ctx, &code->global_code, NULL, NULL, NULL);
81 IActiveScriptSite_OnLeaveScript(ctx->site);
86 static void exec_queued_code(script_ctx_t *ctx)
90 LIST_FOR_EACH_ENTRY(iter, &ctx->code_list, vbscode_t, entry) {
91 if(!iter->global_executed)
92 exec_global_code(ctx, iter);
96 static HRESULT set_ctx_site(VBScript *This)
100 This->ctx->lcid = This->lcid;
102 hres = init_global(This->ctx);
106 IActiveScriptSite_AddRef(This->site);
107 This->ctx->site = This->site;
109 change_state(This, SCRIPTSTATE_INITIALIZED);
113 static void release_script(script_ctx_t *ctx)
115 collect_objects(ctx);
117 while(!list_empty(&ctx->named_items)) {
118 named_item_t *iter = LIST_ENTRY(list_head(&ctx->named_items), named_item_t, entry);
120 list_remove(&iter->entry);
122 IDispatch_Release(iter->disp);
123 heap_free(iter->name);
127 if(ctx->host_global) {
128 IDispatch_Release(ctx->host_global);
129 ctx->host_global = NULL;
133 IInternetHostSecurityManager_Release(ctx->secmgr);
138 IActiveScriptSite_Release(ctx->site);
143 IDispatchEx_Release(&ctx->err_obj->IDispatchEx_iface);
147 if(ctx->global_obj) {
148 IDispatchEx_Release(&ctx->global_obj->IDispatchEx_iface);
149 ctx->global_obj = NULL;
152 if(ctx->script_obj) {
153 IDispatchEx_Release(&ctx->script_obj->IDispatchEx_iface);
154 ctx->script_obj = NULL;
157 vbsheap_free(&ctx->heap);
158 vbsheap_init(&ctx->heap);
161 static void destroy_script(script_ctx_t *ctx)
163 while(!list_empty(&ctx->code_list))
164 release_vbscode(LIST_ENTRY(list_head(&ctx->code_list), vbscode_t, entry));
170 static void decrease_state(VBScript *This, SCRIPTSTATE state)
172 switch(This->state) {
173 case SCRIPTSTATE_CONNECTED:
174 change_state(This, SCRIPTSTATE_DISCONNECTED);
175 if(state == SCRIPTSTATE_DISCONNECTED)
178 case SCRIPTSTATE_STARTED:
179 case SCRIPTSTATE_DISCONNECTED:
180 if(This->state == SCRIPTSTATE_DISCONNECTED)
181 change_state(This, SCRIPTSTATE_INITIALIZED);
182 if(state == SCRIPTSTATE_INITIALIZED)
185 case SCRIPTSTATE_INITIALIZED:
186 case SCRIPTSTATE_UNINITIALIZED:
187 change_state(This, state);
190 IActiveScriptSite_Release(This->site);
195 release_script(This->ctx);
199 case SCRIPTSTATE_CLOSED:
206 static inline VBScript *impl_from_IActiveScript(IActiveScript *iface)
208 return CONTAINING_RECORD(iface, VBScript, IActiveScript_iface);
211 static HRESULT WINAPI VBScript_QueryInterface(IActiveScript *iface, REFIID riid, void **ppv)
213 VBScript *This = impl_from_IActiveScript(iface);
215 if(IsEqualGUID(riid, &IID_IUnknown)) {
216 TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
217 *ppv = &This->IActiveScript_iface;
218 }else if(IsEqualGUID(riid, &IID_IActiveScript)) {
219 TRACE("(%p)->(IID_IActiveScript %p)\n", This, ppv);
220 *ppv = &This->IActiveScript_iface;
221 }else if(IsEqualGUID(riid, &IID_IActiveScriptParse)) {
222 TRACE("(%p)->(IID_IActiveScriptParse %p)\n", This, ppv);
223 *ppv = &This->IActiveScriptParse_iface;
224 }else if(IsEqualGUID(riid, &IID_IObjectSafety)) {
225 TRACE("(%p)->(IID_IObjectSafety %p)\n", This, ppv);
226 *ppv = &This->IObjectSafety_iface;
228 FIXME("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
230 return E_NOINTERFACE;
233 IUnknown_AddRef((IUnknown*)*ppv);
237 static ULONG WINAPI VBScript_AddRef(IActiveScript *iface)
239 VBScript *This = impl_from_IActiveScript(iface);
240 LONG ref = InterlockedIncrement(&This->ref);
242 TRACE("(%p) ref=%d\n", This, ref);
247 static ULONG WINAPI VBScript_Release(IActiveScript *iface)
249 VBScript *This = impl_from_IActiveScript(iface);
250 LONG ref = InterlockedDecrement(&This->ref);
252 TRACE("(%p) ref=%d\n", iface, ref);
256 decrease_state(This, SCRIPTSTATE_CLOSED);
257 destroy_script(This->ctx);
261 IActiveScriptSite_Release(This->site);
268 static HRESULT WINAPI VBScript_SetScriptSite(IActiveScript *iface, IActiveScriptSite *pass)
270 VBScript *This = impl_from_IActiveScript(iface);
274 TRACE("(%p)->(%p)\n", This, pass);
282 if(InterlockedCompareExchange(&This->thread_id, GetCurrentThreadId(), 0))
286 IActiveScriptSite_AddRef(This->site);
288 hres = IActiveScriptSite_GetLCID(This->site, &lcid);
292 return This->ctx ? set_ctx_site(This) : S_OK;
295 static HRESULT WINAPI VBScript_GetScriptSite(IActiveScript *iface, REFIID riid,
298 VBScript *This = impl_from_IActiveScript(iface);
299 FIXME("(%p)->()\n", This);
303 static HRESULT WINAPI VBScript_SetScriptState(IActiveScript *iface, SCRIPTSTATE ss)
305 VBScript *This = impl_from_IActiveScript(iface);
307 TRACE("(%p)->(%d)\n", This, ss);
309 if(This->thread_id && GetCurrentThreadId() != This->thread_id)
312 if(ss == SCRIPTSTATE_UNINITIALIZED) {
313 if(This->state == SCRIPTSTATE_CLOSED)
316 decrease_state(This, SCRIPTSTATE_UNINITIALIZED);
324 case SCRIPTSTATE_STARTED:
325 case SCRIPTSTATE_CONNECTED: /* FIXME */
326 if(This->state == SCRIPTSTATE_CLOSED)
329 exec_queued_code(This->ctx);
331 case SCRIPTSTATE_INITIALIZED:
332 FIXME("unimplemented SCRIPTSTATE_INITIALIZED\n");
335 FIXME("unimplemented state %d\n", ss);
339 change_state(This, ss);
343 static HRESULT WINAPI VBScript_GetScriptState(IActiveScript *iface, SCRIPTSTATE *pssState)
345 VBScript *This = impl_from_IActiveScript(iface);
347 TRACE("(%p)->(%p)\n", This, pssState);
352 if(This->thread_id && This->thread_id != GetCurrentThreadId())
355 *pssState = This->state;
359 static HRESULT WINAPI VBScript_Close(IActiveScript *iface)
361 VBScript *This = impl_from_IActiveScript(iface);
363 TRACE("(%p)->()\n", This);
365 if(This->thread_id && This->thread_id != GetCurrentThreadId())
368 decrease_state(This, SCRIPTSTATE_CLOSED);
372 static HRESULT WINAPI VBScript_AddNamedItem(IActiveScript *iface, LPCOLESTR pstrName, DWORD dwFlags)
374 VBScript *This = impl_from_IActiveScript(iface);
376 IDispatch *disp = NULL;
379 TRACE("(%p)->(%s %x)\n", This, debugstr_w(pstrName), dwFlags);
381 if(This->thread_id != GetCurrentThreadId() || !This->ctx || This->state == SCRIPTSTATE_CLOSED)
384 if(dwFlags & SCRIPTITEM_GLOBALMEMBERS) {
387 hres = IActiveScriptSite_GetItemInfo(This->site, pstrName, SCRIPTINFO_IUNKNOWN, &unk, NULL);
389 WARN("GetItemInfo failed: %08x\n", hres);
393 hres = IUnknown_QueryInterface(unk, &IID_IDispatch, (void**)&disp);
394 IUnknown_Release(unk);
396 WARN("object does not implement IDispatch\n");
400 if(This->ctx->host_global)
401 IDispatch_Release(This->ctx->host_global);
402 IDispatch_AddRef(disp);
403 This->ctx->host_global = disp;
406 item = heap_alloc(sizeof(*item));
409 IDispatch_Release(disp);
410 return E_OUTOFMEMORY;
414 item->flags = dwFlags;
415 item->name = heap_strdupW(pstrName);
418 IDispatch_Release(disp);
420 return E_OUTOFMEMORY;
423 list_add_tail(&This->ctx->named_items, &item->entry);
427 static HRESULT WINAPI VBScript_AddTypeLib(IActiveScript *iface, REFGUID rguidTypeLib,
428 DWORD dwMajor, DWORD dwMinor, DWORD dwFlags)
430 VBScript *This = impl_from_IActiveScript(iface);
431 FIXME("(%p)->()\n", This);
435 static HRESULT WINAPI VBScript_GetScriptDispatch(IActiveScript *iface, LPCOLESTR pstrItemName, IDispatch **ppdisp)
437 VBScript *This = impl_from_IActiveScript(iface);
439 TRACE("(%p)->(%p)\n", This, ppdisp);
444 if(This->thread_id != GetCurrentThreadId() || !This->ctx || !This->ctx->script_obj) {
449 *ppdisp = (IDispatch*)&This->ctx->script_obj->IDispatchEx_iface;
450 IDispatch_AddRef(*ppdisp);
454 static HRESULT WINAPI VBScript_GetCurrentScriptThreadID(IActiveScript *iface,
455 SCRIPTTHREADID *pstridThread)
457 VBScript *This = impl_from_IActiveScript(iface);
458 FIXME("(%p)->()\n", This);
462 static HRESULT WINAPI VBScript_GetScriptThreadID(IActiveScript *iface,
463 DWORD dwWin32ThreadId, SCRIPTTHREADID *pstidThread)
465 VBScript *This = impl_from_IActiveScript(iface);
466 FIXME("(%p)->()\n", This);
470 static HRESULT WINAPI VBScript_GetScriptThreadState(IActiveScript *iface,
471 SCRIPTTHREADID stidThread, SCRIPTTHREADSTATE *pstsState)
473 VBScript *This = impl_from_IActiveScript(iface);
474 FIXME("(%p)->()\n", This);
478 static HRESULT WINAPI VBScript_InterruptScriptThread(IActiveScript *iface,
479 SCRIPTTHREADID stidThread, const EXCEPINFO *pexcepinfo, DWORD dwFlags)
481 VBScript *This = impl_from_IActiveScript(iface);
482 FIXME("(%p)->()\n", This);
486 static HRESULT WINAPI VBScript_Clone(IActiveScript *iface, IActiveScript **ppscript)
488 VBScript *This = impl_from_IActiveScript(iface);
489 FIXME("(%p)->()\n", This);
493 static const IActiveScriptVtbl VBScriptVtbl = {
494 VBScript_QueryInterface,
497 VBScript_SetScriptSite,
498 VBScript_GetScriptSite,
499 VBScript_SetScriptState,
500 VBScript_GetScriptState,
502 VBScript_AddNamedItem,
504 VBScript_GetScriptDispatch,
505 VBScript_GetCurrentScriptThreadID,
506 VBScript_GetScriptThreadID,
507 VBScript_GetScriptThreadState,
508 VBScript_InterruptScriptThread,
512 static inline VBScript *impl_from_IActiveScriptParse(IActiveScriptParse *iface)
514 return CONTAINING_RECORD(iface, VBScript, IActiveScriptParse_iface);
517 static HRESULT WINAPI VBScriptParse_QueryInterface(IActiveScriptParse *iface, REFIID riid, void **ppv)
519 VBScript *This = impl_from_IActiveScriptParse(iface);
520 return IActiveScript_QueryInterface(&This->IActiveScript_iface, riid, ppv);
523 static ULONG WINAPI VBScriptParse_AddRef(IActiveScriptParse *iface)
525 VBScript *This = impl_from_IActiveScriptParse(iface);
526 return IActiveScript_AddRef(&This->IActiveScript_iface);
529 static ULONG WINAPI VBScriptParse_Release(IActiveScriptParse *iface)
531 VBScript *This = impl_from_IActiveScriptParse(iface);
532 return IActiveScript_Release(&This->IActiveScript_iface);
535 static HRESULT WINAPI VBScriptParse_InitNew(IActiveScriptParse *iface)
537 VBScript *This = impl_from_IActiveScriptParse(iface);
538 script_ctx_t *ctx, *old_ctx;
540 TRACE("(%p)\n", This);
545 ctx = heap_alloc_zero(sizeof(script_ctx_t));
547 return E_OUTOFMEMORY;
549 ctx->safeopt = This->safeopt;
550 vbsheap_init(&ctx->heap);
551 list_init(&ctx->objects);
552 list_init(&ctx->code_list);
553 list_init(&ctx->named_items);
555 old_ctx = InterlockedCompareExchangePointer((void**)&This->ctx, ctx, NULL);
561 return This->site ? set_ctx_site(This) : S_OK;
564 static HRESULT WINAPI VBScriptParse_AddScriptlet(IActiveScriptParse *iface,
565 LPCOLESTR pstrDefaultName, LPCOLESTR pstrCode, LPCOLESTR pstrItemName,
566 LPCOLESTR pstrSubItemName, LPCOLESTR pstrEventName, LPCOLESTR pstrDelimiter,
567 CTXARG_T dwSourceContextCookie, ULONG ulStartingLineNumber, DWORD dwFlags,
568 BSTR *pbstrName, EXCEPINFO *pexcepinfo)
570 VBScript *This = impl_from_IActiveScriptParse(iface);
571 FIXME("(%p)->(%s %s %s %s %s %s %s %u %x %p %p)\n", This, debugstr_w(pstrDefaultName),
572 debugstr_w(pstrCode), debugstr_w(pstrItemName), debugstr_w(pstrSubItemName),
573 debugstr_w(pstrEventName), debugstr_w(pstrDelimiter), wine_dbgstr_longlong(dwSourceContextCookie),
574 ulStartingLineNumber, dwFlags, pbstrName, pexcepinfo);
578 static HRESULT WINAPI VBScriptParse_ParseScriptText(IActiveScriptParse *iface,
579 LPCOLESTR pstrCode, LPCOLESTR pstrItemName, IUnknown *punkContext,
580 LPCOLESTR pstrDelimiter, CTXARG_T dwSourceContextCookie, ULONG ulStartingLine,
581 DWORD dwFlags, VARIANT *pvarResult, EXCEPINFO *pexcepinfo)
583 VBScript *This = impl_from_IActiveScriptParse(iface);
587 TRACE("(%p)->(%s %s %p %s %s %u %x %p %p)\n", This, debugstr_w(pstrCode),
588 debugstr_w(pstrItemName), punkContext, debugstr_w(pstrDelimiter),
589 wine_dbgstr_longlong(dwSourceContextCookie), ulStartingLine, dwFlags, pvarResult, pexcepinfo);
591 if(This->thread_id != GetCurrentThreadId() || This->state == SCRIPTSTATE_CLOSED)
594 hres = compile_script(This->ctx, pstrCode, &code);
598 return is_started(This) ? exec_global_code(This->ctx, code) : S_OK;
601 static const IActiveScriptParseVtbl VBScriptParseVtbl = {
602 VBScriptParse_QueryInterface,
603 VBScriptParse_AddRef,
604 VBScriptParse_Release,
605 VBScriptParse_InitNew,
606 VBScriptParse_AddScriptlet,
607 VBScriptParse_ParseScriptText
610 static inline VBScript *impl_from_IObjectSafety(IObjectSafety *iface)
612 return CONTAINING_RECORD(iface, VBScript, IObjectSafety_iface);
615 static HRESULT WINAPI VBScriptSafety_QueryInterface(IObjectSafety *iface, REFIID riid, void **ppv)
617 VBScript *This = impl_from_IObjectSafety(iface);
618 return IActiveScript_QueryInterface(&This->IActiveScript_iface, riid, ppv);
621 static ULONG WINAPI VBScriptSafety_AddRef(IObjectSafety *iface)
623 VBScript *This = impl_from_IObjectSafety(iface);
624 return IActiveScript_AddRef(&This->IActiveScript_iface);
627 static ULONG WINAPI VBScriptSafety_Release(IObjectSafety *iface)
629 VBScript *This = impl_from_IObjectSafety(iface);
630 return IActiveScript_Release(&This->IActiveScript_iface);
633 #define SUPPORTED_OPTIONS (INTERFACESAFE_FOR_UNTRUSTED_DATA|INTERFACE_USES_DISPEX|INTERFACE_USES_SECURITY_MANAGER)
635 static HRESULT WINAPI VBScriptSafety_GetInterfaceSafetyOptions(IObjectSafety *iface, REFIID riid,
636 DWORD *pdwSupportedOptions, DWORD *pdwEnabledOptions)
638 VBScript *This = impl_from_IObjectSafety(iface);
640 TRACE("(%p)->(%s %p %p)\n", This, debugstr_guid(riid), pdwSupportedOptions, pdwEnabledOptions);
642 if(!pdwSupportedOptions || !pdwEnabledOptions)
645 *pdwSupportedOptions = SUPPORTED_OPTIONS;
646 *pdwEnabledOptions = This->safeopt;
650 static HRESULT WINAPI VBScriptSafety_SetInterfaceSafetyOptions(IObjectSafety *iface, REFIID riid,
651 DWORD dwOptionSetMask, DWORD dwEnabledOptions)
653 VBScript *This = impl_from_IObjectSafety(iface);
655 TRACE("(%p)->(%s %x %x)\n", This, debugstr_guid(riid), dwOptionSetMask, dwEnabledOptions);
657 if(dwOptionSetMask & ~SUPPORTED_OPTIONS)
660 This->safeopt = (dwEnabledOptions & dwOptionSetMask) | (This->safeopt & ~dwOptionSetMask) | INTERFACE_USES_DISPEX;
664 static const IObjectSafetyVtbl VBScriptSafetyVtbl = {
665 VBScriptSafety_QueryInterface,
666 VBScriptSafety_AddRef,
667 VBScriptSafety_Release,
668 VBScriptSafety_GetInterfaceSafetyOptions,
669 VBScriptSafety_SetInterfaceSafetyOptions
672 HRESULT WINAPI VBScriptFactory_CreateInstance(IClassFactory *iface, IUnknown *pUnkOuter, REFIID riid, void **ppv)
677 TRACE("(%p %s %p)\n", pUnkOuter, debugstr_guid(riid), ppv);
679 ret = heap_alloc_zero(sizeof(*ret));
681 return E_OUTOFMEMORY;
683 ret->IActiveScript_iface.lpVtbl = &VBScriptVtbl;
684 ret->IActiveScriptParse_iface.lpVtbl = &VBScriptParseVtbl;
685 ret->IObjectSafety_iface.lpVtbl = &VBScriptSafetyVtbl;
688 ret->state = SCRIPTSTATE_UNINITIALIZED;
689 ret->safeopt = INTERFACE_USES_DISPEX;
691 hres = IActiveScript_QueryInterface(&ret->IActiveScript_iface, riid, ppv);
692 IActiveScript_Release(&ret->IActiveScript_iface);
697 IServiceProvider IServiceProvider_iface;
701 IServiceProvider *sp;
704 static inline AXSite *impl_from_IServiceProvider(IServiceProvider *iface)
706 return CONTAINING_RECORD(iface, AXSite, IServiceProvider_iface);
709 static HRESULT WINAPI AXSite_QueryInterface(IServiceProvider *iface, REFIID riid, void **ppv)
711 AXSite *This = impl_from_IServiceProvider(iface);
713 if(IsEqualGUID(&IID_IUnknown, riid)) {
714 TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
715 *ppv = &This->IServiceProvider_iface;
716 }else if(IsEqualGUID(&IID_IServiceProvider, riid)) {
717 TRACE("(%p)->(IID_IServiceProvider %p)\n", This, ppv);
718 *ppv = &This->IServiceProvider_iface;
720 TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
722 return E_NOINTERFACE;
725 IUnknown_AddRef((IUnknown*)*ppv);
729 static ULONG WINAPI AXSite_AddRef(IServiceProvider *iface)
731 AXSite *This = impl_from_IServiceProvider(iface);
732 LONG ref = InterlockedIncrement(&This->ref);
734 TRACE("(%p) ref=%d\n", This, ref);
739 static ULONG WINAPI AXSite_Release(IServiceProvider *iface)
741 AXSite *This = impl_from_IServiceProvider(iface);
742 LONG ref = InterlockedDecrement(&This->ref);
744 TRACE("(%p) ref=%d\n", This, ref);
752 static HRESULT WINAPI AXSite_QueryService(IServiceProvider *iface,
753 REFGUID guidService, REFIID riid, void **ppv)
755 AXSite *This = impl_from_IServiceProvider(iface);
757 TRACE("(%p)->(%s %s %p)\n", This, debugstr_guid(guidService), debugstr_guid(riid), ppv);
759 return IServiceProvider_QueryService(This->sp, guidService, riid, ppv);
762 static IServiceProviderVtbl AXSiteVtbl = {
763 AXSite_QueryInterface,
769 IUnknown *create_ax_site(script_ctx_t *ctx)
771 IServiceProvider *sp;
775 hres = IActiveScriptSite_QueryInterface(ctx->site, &IID_IServiceProvider, (void**)&sp);
777 ERR("Could not get IServiceProvider iface: %08x\n", hres);
781 ret = heap_alloc(sizeof(*ret));
783 IServiceProvider_Release(sp);
787 ret->IServiceProvider_iface.lpVtbl = &AXSiteVtbl;
791 return (IUnknown*)&ret->IServiceProvider_iface;