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
33 #define IActiveScriptParseProcedure2Vtbl IActiveScriptParseProcedure2_64Vtbl
37 #define CTXARG_T DWORD
38 #define IActiveScriptParseVtbl IActiveScriptParse32Vtbl
39 #define IActiveScriptParseProcedure2Vtbl IActiveScriptParseProcedure2_32Vtbl
44 IActiveScript IActiveScript_iface;
45 IActiveScriptParse IActiveScriptParse_iface;
46 IActiveScriptParseProcedure2 IActiveScriptParseProcedure2_iface;
47 IObjectSafety IObjectSafety_iface;
53 IActiveScriptSite *site;
59 static void change_state(VBScript *This, SCRIPTSTATE state)
61 if(This->state == state)
66 IActiveScriptSite_OnStateChange(This->site, state);
69 static inline BOOL is_started(VBScript *This)
71 return This->state == SCRIPTSTATE_STARTED
72 || This->state == SCRIPTSTATE_CONNECTED
73 || This->state == SCRIPTSTATE_DISCONNECTED;
76 static HRESULT exec_global_code(script_ctx_t *ctx, vbscode_t *code)
80 code->pending_exec = FALSE;
82 IActiveScriptSite_OnEnterScript(ctx->site);
83 hres = exec_script(ctx, &code->main_code, NULL, NULL, NULL);
84 IActiveScriptSite_OnLeaveScript(ctx->site);
89 static void exec_queued_code(script_ctx_t *ctx)
93 LIST_FOR_EACH_ENTRY(iter, &ctx->code_list, vbscode_t, entry) {
94 if(iter->pending_exec)
95 exec_global_code(ctx, iter);
99 static HRESULT set_ctx_site(VBScript *This)
103 This->ctx->lcid = This->lcid;
105 hres = init_global(This->ctx);
109 IActiveScriptSite_AddRef(This->site);
110 This->ctx->site = This->site;
112 change_state(This, SCRIPTSTATE_INITIALIZED);
116 static void release_script(script_ctx_t *ctx)
118 class_desc_t *class_desc;
120 collect_objects(ctx);
122 release_dynamic_vars(ctx->global_vars);
123 ctx->global_vars = NULL;
125 while(!list_empty(&ctx->named_items)) {
126 named_item_t *iter = LIST_ENTRY(list_head(&ctx->named_items), named_item_t, entry);
128 list_remove(&iter->entry);
130 IDispatch_Release(iter->disp);
131 heap_free(iter->name);
136 class_desc = ctx->procs;
137 ctx->procs = class_desc->next;
139 heap_free(class_desc);
142 if(ctx->host_global) {
143 IDispatch_Release(ctx->host_global);
144 ctx->host_global = NULL;
148 IInternetHostSecurityManager_Release(ctx->secmgr);
153 IActiveScriptSite_Release(ctx->site);
158 IDispatchEx_Release(&ctx->err_obj->IDispatchEx_iface);
162 if(ctx->global_obj) {
163 IDispatchEx_Release(&ctx->global_obj->IDispatchEx_iface);
164 ctx->global_obj = NULL;
167 if(ctx->script_obj) {
168 IDispatchEx_Release(&ctx->script_obj->IDispatchEx_iface);
169 ctx->script_obj = NULL;
172 vbsheap_free(&ctx->heap);
173 vbsheap_init(&ctx->heap);
176 static void destroy_script(script_ctx_t *ctx)
178 while(!list_empty(&ctx->code_list))
179 release_vbscode(LIST_ENTRY(list_head(&ctx->code_list), vbscode_t, entry));
185 static void decrease_state(VBScript *This, SCRIPTSTATE state)
187 switch(This->state) {
188 case SCRIPTSTATE_CONNECTED:
189 change_state(This, SCRIPTSTATE_DISCONNECTED);
190 if(state == SCRIPTSTATE_DISCONNECTED)
193 case SCRIPTSTATE_STARTED:
194 case SCRIPTSTATE_DISCONNECTED:
195 if(This->state == SCRIPTSTATE_DISCONNECTED)
196 change_state(This, SCRIPTSTATE_INITIALIZED);
197 if(state == SCRIPTSTATE_INITIALIZED)
200 case SCRIPTSTATE_INITIALIZED:
201 case SCRIPTSTATE_UNINITIALIZED:
202 change_state(This, state);
205 IActiveScriptSite_Release(This->site);
210 release_script(This->ctx);
214 case SCRIPTSTATE_CLOSED:
221 static inline VBScript *impl_from_IActiveScript(IActiveScript *iface)
223 return CONTAINING_RECORD(iface, VBScript, IActiveScript_iface);
226 static HRESULT WINAPI VBScript_QueryInterface(IActiveScript *iface, REFIID riid, void **ppv)
228 VBScript *This = impl_from_IActiveScript(iface);
230 if(IsEqualGUID(riid, &IID_IUnknown)) {
231 TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
232 *ppv = &This->IActiveScript_iface;
233 }else if(IsEqualGUID(riid, &IID_IActiveScript)) {
234 TRACE("(%p)->(IID_IActiveScript %p)\n", This, ppv);
235 *ppv = &This->IActiveScript_iface;
236 }else if(IsEqualGUID(riid, &IID_IActiveScriptParse)) {
237 TRACE("(%p)->(IID_IActiveScriptParse %p)\n", This, ppv);
238 *ppv = &This->IActiveScriptParse_iface;
239 }else if(IsEqualGUID(riid, &IID_IActiveScriptParseProcedure2)) {
240 TRACE("(%p)->(IID_IActiveScriptParseProcedure2 %p)\n", This, ppv);
241 *ppv = &This->IActiveScriptParseProcedure2_iface;
242 }else if(IsEqualGUID(riid, &IID_IObjectSafety)) {
243 TRACE("(%p)->(IID_IObjectSafety %p)\n", This, ppv);
244 *ppv = &This->IObjectSafety_iface;
246 FIXME("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
248 return E_NOINTERFACE;
251 IUnknown_AddRef((IUnknown*)*ppv);
255 static ULONG WINAPI VBScript_AddRef(IActiveScript *iface)
257 VBScript *This = impl_from_IActiveScript(iface);
258 LONG ref = InterlockedIncrement(&This->ref);
260 TRACE("(%p) ref=%d\n", This, ref);
265 static ULONG WINAPI VBScript_Release(IActiveScript *iface)
267 VBScript *This = impl_from_IActiveScript(iface);
268 LONG ref = InterlockedDecrement(&This->ref);
270 TRACE("(%p) ref=%d\n", iface, ref);
274 decrease_state(This, SCRIPTSTATE_CLOSED);
275 destroy_script(This->ctx);
279 IActiveScriptSite_Release(This->site);
286 static HRESULT WINAPI VBScript_SetScriptSite(IActiveScript *iface, IActiveScriptSite *pass)
288 VBScript *This = impl_from_IActiveScript(iface);
292 TRACE("(%p)->(%p)\n", This, pass);
300 if(InterlockedCompareExchange(&This->thread_id, GetCurrentThreadId(), 0))
304 IActiveScriptSite_AddRef(This->site);
306 hres = IActiveScriptSite_GetLCID(This->site, &lcid);
310 return This->ctx ? set_ctx_site(This) : S_OK;
313 static HRESULT WINAPI VBScript_GetScriptSite(IActiveScript *iface, REFIID riid,
316 VBScript *This = impl_from_IActiveScript(iface);
317 FIXME("(%p)->()\n", This);
321 static HRESULT WINAPI VBScript_SetScriptState(IActiveScript *iface, SCRIPTSTATE ss)
323 VBScript *This = impl_from_IActiveScript(iface);
325 TRACE("(%p)->(%d)\n", This, ss);
327 if(This->thread_id && GetCurrentThreadId() != This->thread_id)
330 if(ss == SCRIPTSTATE_UNINITIALIZED) {
331 if(This->state == SCRIPTSTATE_CLOSED)
334 decrease_state(This, SCRIPTSTATE_UNINITIALIZED);
342 case SCRIPTSTATE_STARTED:
343 case SCRIPTSTATE_CONNECTED: /* FIXME */
344 if(This->state == SCRIPTSTATE_CLOSED)
347 exec_queued_code(This->ctx);
349 case SCRIPTSTATE_INITIALIZED:
350 FIXME("unimplemented SCRIPTSTATE_INITIALIZED\n");
353 FIXME("unimplemented state %d\n", ss);
357 change_state(This, ss);
361 static HRESULT WINAPI VBScript_GetScriptState(IActiveScript *iface, SCRIPTSTATE *pssState)
363 VBScript *This = impl_from_IActiveScript(iface);
365 TRACE("(%p)->(%p)\n", This, pssState);
370 if(This->thread_id && This->thread_id != GetCurrentThreadId())
373 *pssState = This->state;
377 static HRESULT WINAPI VBScript_Close(IActiveScript *iface)
379 VBScript *This = impl_from_IActiveScript(iface);
381 TRACE("(%p)->()\n", This);
383 if(This->thread_id && This->thread_id != GetCurrentThreadId())
386 decrease_state(This, SCRIPTSTATE_CLOSED);
390 static HRESULT WINAPI VBScript_AddNamedItem(IActiveScript *iface, LPCOLESTR pstrName, DWORD dwFlags)
392 VBScript *This = impl_from_IActiveScript(iface);
394 IDispatch *disp = NULL;
397 TRACE("(%p)->(%s %x)\n", This, debugstr_w(pstrName), dwFlags);
399 if(This->thread_id != GetCurrentThreadId() || !This->ctx || This->state == SCRIPTSTATE_CLOSED)
402 if(dwFlags & SCRIPTITEM_GLOBALMEMBERS) {
405 hres = IActiveScriptSite_GetItemInfo(This->site, pstrName, SCRIPTINFO_IUNKNOWN, &unk, NULL);
407 WARN("GetItemInfo failed: %08x\n", hres);
411 hres = IUnknown_QueryInterface(unk, &IID_IDispatch, (void**)&disp);
412 IUnknown_Release(unk);
414 WARN("object does not implement IDispatch\n");
418 if(This->ctx->host_global)
419 IDispatch_Release(This->ctx->host_global);
420 IDispatch_AddRef(disp);
421 This->ctx->host_global = disp;
424 item = heap_alloc(sizeof(*item));
427 IDispatch_Release(disp);
428 return E_OUTOFMEMORY;
432 item->flags = dwFlags;
433 item->name = heap_strdupW(pstrName);
436 IDispatch_Release(disp);
438 return E_OUTOFMEMORY;
441 list_add_tail(&This->ctx->named_items, &item->entry);
445 static HRESULT WINAPI VBScript_AddTypeLib(IActiveScript *iface, REFGUID rguidTypeLib,
446 DWORD dwMajor, DWORD dwMinor, DWORD dwFlags)
448 VBScript *This = impl_from_IActiveScript(iface);
449 FIXME("(%p)->()\n", This);
453 static HRESULT WINAPI VBScript_GetScriptDispatch(IActiveScript *iface, LPCOLESTR pstrItemName, IDispatch **ppdisp)
455 VBScript *This = impl_from_IActiveScript(iface);
457 TRACE("(%p)->(%p)\n", This, ppdisp);
462 if(This->thread_id != GetCurrentThreadId() || !This->ctx || !This->ctx->script_obj) {
467 *ppdisp = (IDispatch*)&This->ctx->script_obj->IDispatchEx_iface;
468 IDispatch_AddRef(*ppdisp);
472 static HRESULT WINAPI VBScript_GetCurrentScriptThreadID(IActiveScript *iface,
473 SCRIPTTHREADID *pstridThread)
475 VBScript *This = impl_from_IActiveScript(iface);
476 FIXME("(%p)->()\n", This);
480 static HRESULT WINAPI VBScript_GetScriptThreadID(IActiveScript *iface,
481 DWORD dwWin32ThreadId, SCRIPTTHREADID *pstidThread)
483 VBScript *This = impl_from_IActiveScript(iface);
484 FIXME("(%p)->()\n", This);
488 static HRESULT WINAPI VBScript_GetScriptThreadState(IActiveScript *iface,
489 SCRIPTTHREADID stidThread, SCRIPTTHREADSTATE *pstsState)
491 VBScript *This = impl_from_IActiveScript(iface);
492 FIXME("(%p)->()\n", This);
496 static HRESULT WINAPI VBScript_InterruptScriptThread(IActiveScript *iface,
497 SCRIPTTHREADID stidThread, const EXCEPINFO *pexcepinfo, DWORD dwFlags)
499 VBScript *This = impl_from_IActiveScript(iface);
500 FIXME("(%p)->()\n", This);
504 static HRESULT WINAPI VBScript_Clone(IActiveScript *iface, IActiveScript **ppscript)
506 VBScript *This = impl_from_IActiveScript(iface);
507 FIXME("(%p)->()\n", This);
511 static const IActiveScriptVtbl VBScriptVtbl = {
512 VBScript_QueryInterface,
515 VBScript_SetScriptSite,
516 VBScript_GetScriptSite,
517 VBScript_SetScriptState,
518 VBScript_GetScriptState,
520 VBScript_AddNamedItem,
522 VBScript_GetScriptDispatch,
523 VBScript_GetCurrentScriptThreadID,
524 VBScript_GetScriptThreadID,
525 VBScript_GetScriptThreadState,
526 VBScript_InterruptScriptThread,
530 static inline VBScript *impl_from_IActiveScriptParse(IActiveScriptParse *iface)
532 return CONTAINING_RECORD(iface, VBScript, IActiveScriptParse_iface);
535 static HRESULT WINAPI VBScriptParse_QueryInterface(IActiveScriptParse *iface, REFIID riid, void **ppv)
537 VBScript *This = impl_from_IActiveScriptParse(iface);
538 return IActiveScript_QueryInterface(&This->IActiveScript_iface, riid, ppv);
541 static ULONG WINAPI VBScriptParse_AddRef(IActiveScriptParse *iface)
543 VBScript *This = impl_from_IActiveScriptParse(iface);
544 return IActiveScript_AddRef(&This->IActiveScript_iface);
547 static ULONG WINAPI VBScriptParse_Release(IActiveScriptParse *iface)
549 VBScript *This = impl_from_IActiveScriptParse(iface);
550 return IActiveScript_Release(&This->IActiveScript_iface);
553 static HRESULT WINAPI VBScriptParse_InitNew(IActiveScriptParse *iface)
555 VBScript *This = impl_from_IActiveScriptParse(iface);
556 script_ctx_t *ctx, *old_ctx;
558 TRACE("(%p)\n", This);
563 ctx = heap_alloc_zero(sizeof(script_ctx_t));
565 return E_OUTOFMEMORY;
567 ctx->safeopt = This->safeopt;
568 vbsheap_init(&ctx->heap);
569 list_init(&ctx->objects);
570 list_init(&ctx->code_list);
571 list_init(&ctx->named_items);
573 old_ctx = InterlockedCompareExchangePointer((void**)&This->ctx, ctx, NULL);
579 return This->site ? set_ctx_site(This) : S_OK;
582 static HRESULT WINAPI VBScriptParse_AddScriptlet(IActiveScriptParse *iface,
583 LPCOLESTR pstrDefaultName, LPCOLESTR pstrCode, LPCOLESTR pstrItemName,
584 LPCOLESTR pstrSubItemName, LPCOLESTR pstrEventName, LPCOLESTR pstrDelimiter,
585 CTXARG_T dwSourceContextCookie, ULONG ulStartingLineNumber, DWORD dwFlags,
586 BSTR *pbstrName, EXCEPINFO *pexcepinfo)
588 VBScript *This = impl_from_IActiveScriptParse(iface);
589 FIXME("(%p)->(%s %s %s %s %s %s %s %u %x %p %p)\n", This, debugstr_w(pstrDefaultName),
590 debugstr_w(pstrCode), debugstr_w(pstrItemName), debugstr_w(pstrSubItemName),
591 debugstr_w(pstrEventName), debugstr_w(pstrDelimiter), wine_dbgstr_longlong(dwSourceContextCookie),
592 ulStartingLineNumber, dwFlags, pbstrName, pexcepinfo);
596 static HRESULT WINAPI VBScriptParse_ParseScriptText(IActiveScriptParse *iface,
597 LPCOLESTR pstrCode, LPCOLESTR pstrItemName, IUnknown *punkContext,
598 LPCOLESTR pstrDelimiter, CTXARG_T dwSourceContextCookie, ULONG ulStartingLine,
599 DWORD dwFlags, VARIANT *pvarResult, EXCEPINFO *pexcepinfo)
601 VBScript *This = impl_from_IActiveScriptParse(iface);
605 TRACE("(%p)->(%s %s %p %s %s %u %x %p %p)\n", This, debugstr_w(pstrCode),
606 debugstr_w(pstrItemName), punkContext, debugstr_w(pstrDelimiter),
607 wine_dbgstr_longlong(dwSourceContextCookie), ulStartingLine, dwFlags, pvarResult, pexcepinfo);
609 if(This->thread_id != GetCurrentThreadId() || This->state == SCRIPTSTATE_CLOSED)
612 hres = compile_script(This->ctx, pstrCode, &code);
616 if(!is_started(This)) {
617 code->pending_exec = TRUE;
621 return exec_global_code(This->ctx, code);
624 static const IActiveScriptParseVtbl VBScriptParseVtbl = {
625 VBScriptParse_QueryInterface,
626 VBScriptParse_AddRef,
627 VBScriptParse_Release,
628 VBScriptParse_InitNew,
629 VBScriptParse_AddScriptlet,
630 VBScriptParse_ParseScriptText
633 static inline VBScript *impl_from_IActiveScriptParseProcedure2(IActiveScriptParseProcedure2 *iface)
635 return CONTAINING_RECORD(iface, VBScript, IActiveScriptParseProcedure2_iface);
638 static HRESULT WINAPI VBScriptParseProcedure_QueryInterface(IActiveScriptParseProcedure2 *iface, REFIID riid, void **ppv)
640 VBScript *This = impl_from_IActiveScriptParseProcedure2(iface);
641 return IActiveScript_QueryInterface(&This->IActiveScript_iface, riid, ppv);
644 static ULONG WINAPI VBScriptParseProcedure_AddRef(IActiveScriptParseProcedure2 *iface)
646 VBScript *This = impl_from_IActiveScriptParseProcedure2(iface);
647 return IActiveScript_AddRef(&This->IActiveScript_iface);
650 static ULONG WINAPI VBScriptParseProcedure_Release(IActiveScriptParseProcedure2 *iface)
652 VBScript *This = impl_from_IActiveScriptParseProcedure2(iface);
653 return IActiveScript_Release(&This->IActiveScript_iface);
656 static HRESULT WINAPI VBScriptParseProcedure_ParseProcedureText(IActiveScriptParseProcedure2 *iface,
657 LPCOLESTR pstrCode, LPCOLESTR pstrFormalParams, LPCOLESTR pstrProcedureName,
658 LPCOLESTR pstrItemName, IUnknown *punkContext, LPCOLESTR pstrDelimiter,
659 CTXARG_T dwSourceContextCookie, ULONG ulStartingLineNumber, DWORD dwFlags, IDispatch **ppdisp)
661 VBScript *This = impl_from_IActiveScriptParseProcedure2(iface);
665 TRACE("(%p)->(%s %s %s %s %p %s %s %u %x %p)\n", This, debugstr_w(pstrCode), debugstr_w(pstrFormalParams),
666 debugstr_w(pstrProcedureName), debugstr_w(pstrItemName), punkContext, debugstr_w(pstrDelimiter),
667 wine_dbgstr_longlong(dwSourceContextCookie), ulStartingLineNumber, dwFlags, ppdisp);
669 if(This->thread_id != GetCurrentThreadId() || This->state == SCRIPTSTATE_CLOSED)
672 hres = compile_script(This->ctx, pstrCode, &code);
676 return create_procedure_disp(This->ctx, code, ppdisp);
679 static const IActiveScriptParseProcedure2Vtbl VBScriptParseProcedureVtbl = {
680 VBScriptParseProcedure_QueryInterface,
681 VBScriptParseProcedure_AddRef,
682 VBScriptParseProcedure_Release,
683 VBScriptParseProcedure_ParseProcedureText,
686 static inline VBScript *impl_from_IObjectSafety(IObjectSafety *iface)
688 return CONTAINING_RECORD(iface, VBScript, IObjectSafety_iface);
691 static HRESULT WINAPI VBScriptSafety_QueryInterface(IObjectSafety *iface, REFIID riid, void **ppv)
693 VBScript *This = impl_from_IObjectSafety(iface);
694 return IActiveScript_QueryInterface(&This->IActiveScript_iface, riid, ppv);
697 static ULONG WINAPI VBScriptSafety_AddRef(IObjectSafety *iface)
699 VBScript *This = impl_from_IObjectSafety(iface);
700 return IActiveScript_AddRef(&This->IActiveScript_iface);
703 static ULONG WINAPI VBScriptSafety_Release(IObjectSafety *iface)
705 VBScript *This = impl_from_IObjectSafety(iface);
706 return IActiveScript_Release(&This->IActiveScript_iface);
709 #define SUPPORTED_OPTIONS (INTERFACESAFE_FOR_UNTRUSTED_DATA|INTERFACE_USES_DISPEX|INTERFACE_USES_SECURITY_MANAGER)
711 static HRESULT WINAPI VBScriptSafety_GetInterfaceSafetyOptions(IObjectSafety *iface, REFIID riid,
712 DWORD *pdwSupportedOptions, DWORD *pdwEnabledOptions)
714 VBScript *This = impl_from_IObjectSafety(iface);
716 TRACE("(%p)->(%s %p %p)\n", This, debugstr_guid(riid), pdwSupportedOptions, pdwEnabledOptions);
718 if(!pdwSupportedOptions || !pdwEnabledOptions)
721 *pdwSupportedOptions = SUPPORTED_OPTIONS;
722 *pdwEnabledOptions = This->safeopt;
726 static HRESULT WINAPI VBScriptSafety_SetInterfaceSafetyOptions(IObjectSafety *iface, REFIID riid,
727 DWORD dwOptionSetMask, DWORD dwEnabledOptions)
729 VBScript *This = impl_from_IObjectSafety(iface);
731 TRACE("(%p)->(%s %x %x)\n", This, debugstr_guid(riid), dwOptionSetMask, dwEnabledOptions);
733 if(dwOptionSetMask & ~SUPPORTED_OPTIONS)
736 This->safeopt = (dwEnabledOptions & dwOptionSetMask) | (This->safeopt & ~dwOptionSetMask) | INTERFACE_USES_DISPEX;
740 static const IObjectSafetyVtbl VBScriptSafetyVtbl = {
741 VBScriptSafety_QueryInterface,
742 VBScriptSafety_AddRef,
743 VBScriptSafety_Release,
744 VBScriptSafety_GetInterfaceSafetyOptions,
745 VBScriptSafety_SetInterfaceSafetyOptions
748 HRESULT WINAPI VBScriptFactory_CreateInstance(IClassFactory *iface, IUnknown *pUnkOuter, REFIID riid, void **ppv)
753 TRACE("(%p %s %p)\n", pUnkOuter, debugstr_guid(riid), ppv);
755 ret = heap_alloc_zero(sizeof(*ret));
757 return E_OUTOFMEMORY;
759 ret->IActiveScript_iface.lpVtbl = &VBScriptVtbl;
760 ret->IActiveScriptParse_iface.lpVtbl = &VBScriptParseVtbl;
761 ret->IActiveScriptParseProcedure2_iface.lpVtbl = &VBScriptParseProcedureVtbl;
762 ret->IObjectSafety_iface.lpVtbl = &VBScriptSafetyVtbl;
765 ret->state = SCRIPTSTATE_UNINITIALIZED;
766 ret->safeopt = INTERFACE_USES_DISPEX;
768 hres = IActiveScript_QueryInterface(&ret->IActiveScript_iface, riid, ppv);
769 IActiveScript_Release(&ret->IActiveScript_iface);
774 IServiceProvider IServiceProvider_iface;
778 IServiceProvider *sp;
781 static inline AXSite *impl_from_IServiceProvider(IServiceProvider *iface)
783 return CONTAINING_RECORD(iface, AXSite, IServiceProvider_iface);
786 static HRESULT WINAPI AXSite_QueryInterface(IServiceProvider *iface, REFIID riid, void **ppv)
788 AXSite *This = impl_from_IServiceProvider(iface);
790 if(IsEqualGUID(&IID_IUnknown, riid)) {
791 TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
792 *ppv = &This->IServiceProvider_iface;
793 }else if(IsEqualGUID(&IID_IServiceProvider, riid)) {
794 TRACE("(%p)->(IID_IServiceProvider %p)\n", This, ppv);
795 *ppv = &This->IServiceProvider_iface;
797 TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
799 return E_NOINTERFACE;
802 IUnknown_AddRef((IUnknown*)*ppv);
806 static ULONG WINAPI AXSite_AddRef(IServiceProvider *iface)
808 AXSite *This = impl_from_IServiceProvider(iface);
809 LONG ref = InterlockedIncrement(&This->ref);
811 TRACE("(%p) ref=%d\n", This, ref);
816 static ULONG WINAPI AXSite_Release(IServiceProvider *iface)
818 AXSite *This = impl_from_IServiceProvider(iface);
819 LONG ref = InterlockedDecrement(&This->ref);
821 TRACE("(%p) ref=%d\n", This, ref);
829 static HRESULT WINAPI AXSite_QueryService(IServiceProvider *iface,
830 REFGUID guidService, REFIID riid, void **ppv)
832 AXSite *This = impl_from_IServiceProvider(iface);
834 TRACE("(%p)->(%s %s %p)\n", This, debugstr_guid(guidService), debugstr_guid(riid), ppv);
836 return IServiceProvider_QueryService(This->sp, guidService, riid, ppv);
839 static IServiceProviderVtbl AXSiteVtbl = {
840 AXSite_QueryInterface,
846 IUnknown *create_ax_site(script_ctx_t *ctx)
848 IServiceProvider *sp;
852 hres = IActiveScriptSite_QueryInterface(ctx->site, &IID_IServiceProvider, (void**)&sp);
854 ERR("Could not get IServiceProvider iface: %08x\n", hres);
858 ret = heap_alloc(sizeof(*ret));
860 IServiceProvider_Release(sp);
864 ret->IServiceProvider_iface.lpVtbl = &AXSiteVtbl;
868 return (IUnknown*)&ret->IServiceProvider_iface;