jscript: Added VBArray.dimensions() implementation.
[wine] / dlls / jscript / tests / run.c
1 /*
2  * Copyright 2008 Jacek Caban for CodeWeavers
3  *
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.
8  *
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.
13  *
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
17  */
18
19 #include <stdio.h>
20
21 #define COBJMACROS
22 #define CONST_VTABLE
23
24 #include <ole2.h>
25 #include <dispex.h>
26 #include <activscp.h>
27
28 #include "wine/test.h"
29
30 static const CLSID CLSID_JScript =
31     {0xf414c260,0x6ac0,0x11cf,{0xb6,0xd1,0x00,0xaa,0x00,0xbb,0xbb,0x58}};
32
33 #define DEFINE_EXPECT(func) \
34     static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
35
36 #define SET_EXPECT(func) \
37     expect_ ## func = TRUE
38
39 #define SET_CALLED(func) \
40     called_ ## func = TRUE
41
42 #define CHECK_EXPECT2(func) \
43     do { \
44         ok(expect_ ##func, "unexpected call " #func "\n"); \
45         called_ ## func = TRUE; \
46     }while(0)
47
48 #define CHECK_EXPECT(func) \
49     do { \
50         CHECK_EXPECT2(func); \
51         expect_ ## func = FALSE; \
52     }while(0)
53
54 #define CHECK_CALLED(func) \
55     do { \
56         ok(called_ ## func, "expected " #func "\n"); \
57         expect_ ## func = called_ ## func = FALSE; \
58     }while(0)
59
60 DEFINE_EXPECT(global_propget_d);
61 DEFINE_EXPECT(global_propget_i);
62 DEFINE_EXPECT(global_propput_d);
63 DEFINE_EXPECT(global_propput_i);
64 DEFINE_EXPECT(global_success_d);
65 DEFINE_EXPECT(global_success_i);
66 DEFINE_EXPECT(global_notexists_d);
67 DEFINE_EXPECT(testobj_delete);
68 DEFINE_EXPECT(testobj_value);
69 DEFINE_EXPECT(testobj_prop_d);
70 DEFINE_EXPECT(testobj_noprop_d);
71 DEFINE_EXPECT(GetItemInfo_testVal);
72 DEFINE_EXPECT(ActiveScriptSite_OnScriptError);
73
74 #define DISPID_GLOBAL_TESTPROPGET   0x1000
75 #define DISPID_GLOBAL_TESTPROPPUT   0x1001
76 #define DISPID_GLOBAL_REPORTSUCCESS 0x1002
77 #define DISPID_GLOBAL_TRACE         0x1003
78 #define DISPID_GLOBAL_OK            0x1004
79 #define DISPID_GLOBAL_GETVT         0x1005
80 #define DISPID_GLOBAL_TESTOBJ       0x1006
81 #define DISPID_GLOBAL_NULL_BSTR     0x1007
82 #define DISPID_GLOBAL_NULL_DISP     0x1008
83 #define DISPID_GLOBAL_TESTTHIS      0x1009
84 #define DISPID_GLOBAL_TESTTHIS2     0x100a
85 #define DISPID_GLOBAL_INVOKEVERSION 0x100b
86 #define DISPID_TEST_CREATEARRAY     0x100c
87
88 #define DISPID_TESTOBJ_PROP         0x2000
89
90 static const WCHAR testW[] = {'t','e','s','t',0};
91 static const CHAR testA[] = "test";
92 static const WCHAR test_valW[] = {'t','e','s','t','V','a','l',0};
93 static const CHAR test_valA[] = "testVal";
94
95 static BOOL strict_dispid_check;
96 static const char *test_name = "(null)";
97 static IDispatch *script_disp;
98 static int invoke_version;
99 static IActiveScriptError *script_error;
100
101 static BSTR a2bstr(const char *str)
102 {
103     BSTR ret;
104     int len;
105
106     len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
107     ret = SysAllocStringLen(NULL, len-1);
108     MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
109
110     return ret;
111 }
112
113 static int strcmp_wa(LPCWSTR strw, const char *stra)
114 {
115     CHAR buf[512];
116     WideCharToMultiByte(CP_ACP, 0, strw, -1, buf, sizeof(buf), 0, 0);
117     return lstrcmpA(buf, stra);
118 }
119
120 #define test_grfdex(a,b) _test_grfdex(__LINE__,a,b)
121 static void _test_grfdex(unsigned line, DWORD grfdex, DWORD expect)
122 {
123     expect |= invoke_version << 28;
124     ok_(__FILE__,line)(grfdex == expect, "grfdex = %x, expected %x\n", grfdex, expect);
125 }
126
127 static HRESULT WINAPI DispatchEx_QueryInterface(IDispatchEx *iface, REFIID riid, void **ppv)
128 {
129     *ppv = NULL;
130
131     if(IsEqualGUID(riid, &IID_IUnknown)
132        || IsEqualGUID(riid, &IID_IDispatch)
133        || IsEqualGUID(riid, &IID_IDispatchEx))
134         *ppv = iface;
135     else
136         return E_NOINTERFACE;
137
138     return S_OK;
139 }
140
141 static ULONG WINAPI DispatchEx_AddRef(IDispatchEx *iface)
142 {
143     return 2;
144 }
145
146 static ULONG WINAPI DispatchEx_Release(IDispatchEx *iface)
147 {
148     return 1;
149 }
150
151 static HRESULT WINAPI DispatchEx_GetTypeInfoCount(IDispatchEx *iface, UINT *pctinfo)
152 {
153     ok(0, "unexpected call\n");
154     return E_NOTIMPL;
155 }
156
157 static HRESULT WINAPI DispatchEx_GetTypeInfo(IDispatchEx *iface, UINT iTInfo,
158                                               LCID lcid, ITypeInfo **ppTInfo)
159 {
160     ok(0, "unexpected call\n");
161     return E_NOTIMPL;
162 }
163
164 static HRESULT WINAPI DispatchEx_GetIDsOfNames(IDispatchEx *iface, REFIID riid,
165                                                 LPOLESTR *rgszNames, UINT cNames,
166                                                 LCID lcid, DISPID *rgDispId)
167 {
168     ok(0, "unexpected call\n");
169     return E_NOTIMPL;
170 }
171
172 static HRESULT WINAPI DispatchEx_Invoke(IDispatchEx *iface, DISPID dispIdMember,
173                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
174                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
175 {
176     ok(0, "unexpected call\n");
177     return E_NOTIMPL;
178 }
179
180 static HRESULT WINAPI DispatchEx_DeleteMemberByName(IDispatchEx *iface, BSTR bstrName, DWORD grfdex)
181 {
182     ok(0, "unexpected call %s %x\n", wine_dbgstr_w(bstrName), grfdex);
183     return E_NOTIMPL;
184 }
185
186 static HRESULT WINAPI DispatchEx_DeleteMemberByDispID(IDispatchEx *iface, DISPID id)
187 {
188     ok(0, "unexpected call\n");
189     return E_NOTIMPL;
190 }
191
192 static HRESULT WINAPI DispatchEx_GetMemberProperties(IDispatchEx *iface, DISPID id, DWORD grfdexFetch, DWORD *pgrfdex)
193 {
194     ok(0, "unexpected call\n");
195     return E_NOTIMPL;
196 }
197
198 static HRESULT WINAPI DispatchEx_GetMemberName(IDispatchEx *iface, DISPID id, BSTR *pbstrName)
199 {
200     ok(0, "unexpected call\n");
201     return E_NOTIMPL;
202 }
203
204 static HRESULT WINAPI DispatchEx_GetNextDispID(IDispatchEx *iface, DWORD grfdex, DISPID id, DISPID *pid)
205 {
206     ok(0, "unexpected call\n");
207     return E_NOTIMPL;
208 }
209
210 static HRESULT WINAPI DispatchEx_GetNameSpaceParent(IDispatchEx *iface, IUnknown **ppunk)
211 {
212     ok(0, "unexpected call\n");
213     return E_NOTIMPL;
214 }
215
216 static HRESULT WINAPI testObj_GetDispID(IDispatchEx *iface, BSTR bstrName, DWORD grfdex, DISPID *pid)
217 {
218     if(!strcmp_wa(bstrName, "prop")) {
219         CHECK_EXPECT(testobj_prop_d);
220         test_grfdex(grfdex, fdexNameCaseSensitive);
221         *pid = DISPID_TESTOBJ_PROP;
222         return S_OK;
223     }
224     if(!strcmp_wa(bstrName, "noprop")) {
225         CHECK_EXPECT(testobj_noprop_d);
226         test_grfdex(grfdex, fdexNameCaseSensitive);
227         return DISP_E_UNKNOWNNAME;
228     }
229
230     ok(0, "unexpected name %s\n", wine_dbgstr_w(bstrName));
231     return E_NOTIMPL;
232 }
233
234 static HRESULT WINAPI testObj_InvokeEx(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
235         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
236 {
237     switch(id) {
238     case DISPID_VALUE:
239         CHECK_EXPECT(testobj_value);
240
241         ok(wFlags == INVOKE_PROPERTYGET, "wFlags = %x\n", wFlags);
242         ok(pdp != NULL, "pdp == NULL\n");
243         ok(!pdp->rgvarg, "rgvarg != NULL\n");
244         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
245         ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
246         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
247         ok(pvarRes != NULL, "pvarRes == NULL\n");
248         ok(V_VT(pvarRes) ==  VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
249         ok(pei != NULL, "pei == NULL\n");
250
251         V_VT(pvarRes) = VT_I4;
252         V_I4(pvarRes) = 1;
253         return S_OK;
254     }
255
256     ok(0, "unexpected call %x\n", id);
257     return DISP_E_MEMBERNOTFOUND;
258 }
259
260 static HRESULT WINAPI testObj_DeleteMemberByName(IDispatchEx *iface, BSTR bstrName, DWORD grfdex)
261 {
262     CHECK_EXPECT(testobj_delete);
263
264     ok(!strcmp_wa(bstrName, "deleteTest"), "unexpected name %s\n", wine_dbgstr_w(bstrName));
265     test_grfdex(grfdex, fdexNameCaseSensitive);
266     return S_OK;
267 }
268
269 static IDispatchExVtbl testObjVtbl = {
270     DispatchEx_QueryInterface,
271     DispatchEx_AddRef,
272     DispatchEx_Release,
273     DispatchEx_GetTypeInfoCount,
274     DispatchEx_GetTypeInfo,
275     DispatchEx_GetIDsOfNames,
276     DispatchEx_Invoke,
277     testObj_GetDispID,
278     testObj_InvokeEx,
279     testObj_DeleteMemberByName,
280     DispatchEx_DeleteMemberByDispID,
281     DispatchEx_GetMemberProperties,
282     DispatchEx_GetMemberName,
283     DispatchEx_GetNextDispID,
284     DispatchEx_GetNameSpaceParent
285 };
286
287 static IDispatchEx testObj = { &testObjVtbl };
288
289 static HRESULT WINAPI Global_GetDispID(IDispatchEx *iface, BSTR bstrName, DWORD grfdex, DISPID *pid)
290 {
291     if(!strcmp_wa(bstrName, "ok")) {
292         test_grfdex(grfdex, fdexNameCaseSensitive);
293         *pid = DISPID_GLOBAL_OK;
294         return S_OK;
295     }
296     if(!strcmp_wa(bstrName, "trace")) {
297         test_grfdex(grfdex, fdexNameCaseSensitive);
298         *pid = DISPID_GLOBAL_TRACE;
299         return S_OK;
300     }
301     if(!strcmp_wa(bstrName, "reportSuccess")) {
302         CHECK_EXPECT(global_success_d);
303         test_grfdex(grfdex, fdexNameCaseSensitive);
304         *pid = DISPID_GLOBAL_REPORTSUCCESS;
305         return S_OK;
306     }
307     if(!strcmp_wa(bstrName, "testPropGet")) {
308         CHECK_EXPECT(global_propget_d);
309         test_grfdex(grfdex, fdexNameCaseSensitive);
310         *pid = DISPID_GLOBAL_TESTPROPGET;
311         return S_OK;
312     }
313     if(!strcmp_wa(bstrName, "testPropPut")) {
314         CHECK_EXPECT(global_propput_d);
315         test_grfdex(grfdex, fdexNameCaseSensitive);
316         *pid = DISPID_GLOBAL_TESTPROPPUT;
317         return S_OK;
318     }
319     if(!strcmp_wa(bstrName, "getVT")) {
320         test_grfdex(grfdex, fdexNameCaseSensitive);
321         *pid = DISPID_GLOBAL_GETVT;
322         return S_OK;
323     }
324     if(!strcmp_wa(bstrName, "testObj")) {
325         test_grfdex(grfdex, fdexNameCaseSensitive);
326         *pid = DISPID_GLOBAL_TESTOBJ;
327         return S_OK;
328     }
329     if(!strcmp_wa(bstrName, "createNullBSTR")) {
330         *pid = DISPID_GLOBAL_NULL_BSTR;
331         return S_OK;
332     }
333     if(!strcmp_wa(bstrName, "nullDisp")) {
334         *pid = DISPID_GLOBAL_NULL_DISP;
335         return S_OK;
336     }
337     if(!strcmp_wa(bstrName, "notExists")) {
338         CHECK_EXPECT(global_notexists_d);
339         test_grfdex(grfdex, fdexNameCaseSensitive);
340         return DISP_E_UNKNOWNNAME;
341     }
342
343     if(!strcmp_wa(bstrName, "testThis")) {
344         test_grfdex(grfdex, fdexNameCaseSensitive);
345         *pid = DISPID_GLOBAL_TESTTHIS;
346         return S_OK;
347     }
348
349     if(!strcmp_wa(bstrName, "testThis2")) {
350         test_grfdex(grfdex, fdexNameCaseSensitive);
351         *pid = DISPID_GLOBAL_TESTTHIS2;
352         return S_OK;
353     }
354
355     if(!strcmp_wa(bstrName, "invokeVersion")) {
356         test_grfdex(grfdex, fdexNameCaseSensitive);
357         *pid = DISPID_GLOBAL_INVOKEVERSION;
358         return S_OK;
359     }
360     if(!strcmp_wa(bstrName, "createArray")) {
361         test_grfdex(grfdex, fdexNameCaseSensitive);
362         *pid = DISPID_TEST_CREATEARRAY;
363         return S_OK;
364     }
365
366     if(strict_dispid_check)
367         ok(0, "unexpected call %s\n", wine_dbgstr_w(bstrName));
368     return DISP_E_UNKNOWNNAME;
369 }
370
371 static HRESULT WINAPI Global_InvokeEx(IDispatchEx *iface, DISPID id, LCID lcid, WORD wFlags, DISPPARAMS *pdp,
372         VARIANT *pvarRes, EXCEPINFO *pei, IServiceProvider *pspCaller)
373 {
374     switch(id) {
375     case DISPID_GLOBAL_OK:
376         ok(wFlags == INVOKE_FUNC || wFlags == (INVOKE_FUNC|INVOKE_PROPERTYGET), "wFlags = %x\n", wFlags);
377         ok(pdp != NULL, "pdp == NULL\n");
378         ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
379         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
380         ok(pdp->cArgs == 2, "cArgs = %d\n", pdp->cArgs);
381         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
382         if(wFlags & INVOKE_PROPERTYGET)
383             ok(pvarRes != NULL, "pvarRes == NULL\n");
384         else
385             ok(!pvarRes, "pvarRes != NULL\n");
386         ok(pei != NULL, "pei == NULL\n");
387
388         ok(V_VT(pdp->rgvarg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
389         ok(V_VT(pdp->rgvarg+1) == VT_BOOL, "V_VT(psp->rgvargs+1) = %d\n", V_VT(pdp->rgvarg));
390         ok(V_BOOL(pdp->rgvarg+1), "%s: %s\n", test_name, wine_dbgstr_w(V_BSTR(pdp->rgvarg)));
391
392         return S_OK;
393
394      case DISPID_GLOBAL_TRACE:
395         ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
396         ok(pdp != NULL, "pdp == NULL\n");
397         ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
398         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
399         ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
400         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
401         ok(!pvarRes, "pvarRes != NULL\n");
402         ok(pei != NULL, "pei == NULL\n");
403
404         ok(V_VT(pdp->rgvarg) == VT_BSTR, "V_VT(psp->rgvargs) = %d\n", V_VT(pdp->rgvarg));
405         if(V_VT(pdp->rgvarg) == VT_BSTR)
406             trace("%s: %s\n", test_name, wine_dbgstr_w(V_BSTR(pdp->rgvarg)));
407
408         return S_OK;
409
410     case DISPID_GLOBAL_REPORTSUCCESS:
411         CHECK_EXPECT(global_success_i);
412
413         ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
414         ok(pdp != NULL, "pdp == NULL\n");
415         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
416         ok(pdp->cArgs == 0, "cArgs = %d\n", pdp->cArgs);
417         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
418         ok(!pvarRes, "pvarRes != NULL\n");
419         ok(pei != NULL, "pei == NULL\n");
420
421         return S_OK;
422
423      case DISPID_GLOBAL_TESTPROPGET:
424         CHECK_EXPECT(global_propget_i);
425
426         ok(wFlags == INVOKE_PROPERTYGET, "wFlags = %x\n", wFlags);
427         ok(pdp != NULL, "pdp == NULL\n");
428         ok(!pdp->rgvarg, "rgvarg != NULL\n");
429         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
430         ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
431         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
432         ok(pvarRes != NULL, "pvarRes == NULL\n");
433         ok(V_VT(pvarRes) ==  VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
434         ok(pei != NULL, "pei == NULL\n");
435
436         V_VT(pvarRes) = VT_I4;
437         V_I4(pvarRes) = 1;
438
439         return S_OK;
440
441     case DISPID_GLOBAL_TESTPROPPUT:
442         CHECK_EXPECT(global_propput_i);
443
444         ok(wFlags == INVOKE_PROPERTYPUT, "wFlags = %x\n", wFlags);
445         ok(pdp != NULL, "pdp == NULL\n");
446         ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
447         ok(pdp->rgdispidNamedArgs != NULL, "rgdispidNamedArgs == NULL\n");
448         ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
449         ok(pdp->cNamedArgs == 1, "cNamedArgs = %d\n", pdp->cNamedArgs);
450         ok(pdp->rgdispidNamedArgs[0] == DISPID_PROPERTYPUT, "pdp->rgdispidNamedArgs[0] = %d\n", pdp->rgdispidNamedArgs[0]);
451         ok(!pvarRes, "pvarRes != NULL\n");
452
453         ok(V_VT(pdp->rgvarg) == VT_I4, "V_VT(pdp->rgvarg)=%d\n", V_VT(pdp->rgvarg));
454         ok(V_I4(pdp->rgvarg) == 1, "V_I4(pdp->rgvarg)=%d\n", V_I4(pdp->rgvarg));
455         return S_OK;
456
457      case DISPID_GLOBAL_GETVT:
458         ok(pdp != NULL, "pdp == NULL\n");
459         ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
460         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
461         ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
462         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
463         ok(pvarRes != NULL, "pvarRes == NULL\n");
464         ok(V_VT(pvarRes) ==  VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
465         ok(pei != NULL, "pei == NULL\n");
466
467         V_VT(pvarRes) = VT_BSTR;
468         switch(V_VT(pdp->rgvarg)) {
469         case VT_EMPTY:
470             V_BSTR(pvarRes) = a2bstr("VT_EMPTY");
471             break;
472         case VT_NULL:
473             V_BSTR(pvarRes) = a2bstr("VT_NULL");
474             break;
475         case VT_I4:
476             V_BSTR(pvarRes) = a2bstr("VT_I4");
477             break;
478         case VT_R8:
479             V_BSTR(pvarRes) = a2bstr("VT_R8");
480             break;
481         case VT_BSTR:
482             V_BSTR(pvarRes) = a2bstr("VT_BSTR");
483             break;
484         case VT_DISPATCH:
485             V_BSTR(pvarRes) = a2bstr("VT_DISPATCH");
486             break;
487         case VT_BOOL:
488             V_BSTR(pvarRes) = a2bstr("VT_BOOL");
489             break;
490         default:
491             ok(0, "unknown vt %d\n", V_VT(pdp->rgvarg));
492             return E_FAIL;
493         }
494
495         return S_OK;
496
497     case DISPID_GLOBAL_TESTOBJ:
498         ok(wFlags == INVOKE_PROPERTYGET, "wFlags = %x\n", wFlags);
499         ok(pdp != NULL, "pdp == NULL\n");
500         ok(!pdp->rgvarg, "rgvarg != NULL\n");
501         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
502         ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
503         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
504         ok(pvarRes != NULL, "pvarRes == NULL\n");
505         ok(V_VT(pvarRes) ==  VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
506         ok(pei != NULL, "pei == NULL\n");
507
508         V_VT(pvarRes) = VT_DISPATCH;
509         V_DISPATCH(pvarRes) = (IDispatch*)&testObj;
510         return S_OK;
511
512     case DISPID_GLOBAL_NULL_BSTR:
513         if(pvarRes) {
514             V_VT(pvarRes) = VT_BSTR;
515             V_BSTR(pvarRes) = NULL;
516         }
517         return S_OK;
518
519     case DISPID_GLOBAL_NULL_DISP:
520         ok(wFlags == INVOKE_PROPERTYGET, "wFlags = %x\n", wFlags);
521         ok(pdp != NULL, "pdp == NULL\n");
522         ok(!pdp->rgvarg, "rgvarg != NULL\n");
523         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
524         ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
525         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
526         ok(pvarRes != NULL, "pvarRes == NULL\n");
527         ok(V_VT(pvarRes) ==  VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
528         ok(pei != NULL, "pei == NULL\n");
529
530         V_VT(pvarRes) = VT_DISPATCH;
531         V_DISPATCH(pvarRes) = NULL;
532         return S_OK;
533
534     case DISPID_GLOBAL_TESTTHIS:
535         ok(pdp != NULL, "pdp == NULL\n");
536         ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
537         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
538         ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
539         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
540         ok(pvarRes == NULL, "pvarRes != NULL\n");
541         ok(pei != NULL, "pei == NULL\n");
542
543         ok(V_VT(pdp->rgvarg) == VT_DISPATCH, "V_VT(arg) = %d\n", V_VT(pdp->rgvarg));
544         ok(V_DISPATCH(pdp->rgvarg) == (IDispatch*)iface, "disp != iface\n");
545
546         return S_OK;
547
548     case DISPID_GLOBAL_TESTTHIS2:
549         ok(pdp != NULL, "pdp == NULL\n");
550         ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
551         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
552         ok(pdp->cArgs == 1, "cArgs = %d\n", pdp->cArgs);
553         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
554         ok(pvarRes == NULL, "pvarRes != NULL\n");
555         ok(pei != NULL, "pei == NULL\n");
556
557         ok(V_VT(pdp->rgvarg) == VT_DISPATCH, "V_VT(arg) = %d\n", V_VT(pdp->rgvarg));
558         ok(V_DISPATCH(pdp->rgvarg) != (IDispatch*)iface, "disp == iface\n");
559         ok(V_DISPATCH(pdp->rgvarg) == script_disp, "disp != script_disp\n");
560
561         return S_OK;
562
563      case DISPID_GLOBAL_INVOKEVERSION:
564         ok(wFlags == INVOKE_PROPERTYGET, "wFlags = %x\n", wFlags);
565         ok(pdp != NULL, "pdp == NULL\n");
566         ok(!pdp->rgvarg, "rgvarg != NULL\n");
567         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
568         ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
569         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
570         ok(pvarRes != NULL, "pvarRes == NULL\n");
571         ok(V_VT(pvarRes) ==  VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
572         ok(pei != NULL, "pei == NULL\n");
573
574         V_VT(pvarRes) = VT_I4;
575         V_I4(pvarRes) = invoke_version;
576
577         return S_OK;
578
579     case DISPID_TEST_CREATEARRAY: {
580         SAFEARRAYBOUND bound[2];
581         VARIANT *data;
582         int i,j;
583
584         ok(wFlags == INVOKE_FUNC, "wFlags = %x\n", wFlags);
585         ok(pdp != NULL, "pdp == NULL\n");
586         todo_wine ok(pdp->rgvarg != NULL, "rgvarg == NULL\n");
587         ok(!pdp->rgdispidNamedArgs, "rgdispidNamedArgs != NULL\n");
588         ok(!pdp->cArgs, "cArgs = %d\n", pdp->cArgs);
589         ok(!pdp->cNamedArgs, "cNamedArgs = %d\n", pdp->cNamedArgs);
590         ok(pvarRes != NULL, "pvarRes == NULL\n");
591         ok(V_VT(pvarRes) ==  VT_EMPTY, "V_VT(pvarRes) = %d\n", V_VT(pvarRes));
592         ok(pei != NULL, "pei == NULL\n");
593
594         bound[0].lLbound = 0;
595         bound[0].cElements = 5;
596         bound[1].lLbound = 2;
597         bound[1].cElements = 2;
598
599         V_VT(pvarRes) = VT_ARRAY|VT_VARIANT;
600         V_ARRAY(pvarRes) = SafeArrayCreate(VT_VARIANT, 2, bound);
601
602         SafeArrayAccessData(V_ARRAY(pvarRes), (void**)&data);
603         for(i=0; i<5; i++) {
604             for(j=2; j<4; j++) {
605                 V_VT(data) = VT_I4;
606                 V_I4(data) = i*10+j;
607                 data++;
608             }
609         }
610         SafeArrayUnaccessData(V_ARRAY(pvarRes));
611
612         return S_OK;
613     }
614     }
615
616     ok(0, "unexpected call %x\n", id);
617     return DISP_E_MEMBERNOTFOUND;
618 }
619
620 static IDispatchExVtbl GlobalVtbl = {
621     DispatchEx_QueryInterface,
622     DispatchEx_AddRef,
623     DispatchEx_Release,
624     DispatchEx_GetTypeInfoCount,
625     DispatchEx_GetTypeInfo,
626     DispatchEx_GetIDsOfNames,
627     DispatchEx_Invoke,
628     Global_GetDispID,
629     Global_InvokeEx,
630     DispatchEx_DeleteMemberByName,
631     DispatchEx_DeleteMemberByDispID,
632     DispatchEx_GetMemberProperties,
633     DispatchEx_GetMemberName,
634     DispatchEx_GetNextDispID,
635     DispatchEx_GetNameSpaceParent
636 };
637
638 static IDispatchEx Global = { &GlobalVtbl };
639
640 static HRESULT WINAPI ActiveScriptSite_QueryInterface(IActiveScriptSite *iface, REFIID riid, void **ppv)
641 {
642     *ppv = NULL;
643
644     if(IsEqualGUID(&IID_IUnknown, riid))
645         *ppv = iface;
646     else if(IsEqualGUID(&IID_IActiveScriptSite, riid))
647         *ppv = iface;
648     else
649         return E_NOINTERFACE;
650
651     IUnknown_AddRef((IUnknown*)*ppv);
652     return S_OK;
653 }
654
655 static ULONG WINAPI ActiveScriptSite_AddRef(IActiveScriptSite *iface)
656 {
657     return 2;
658 }
659
660 static ULONG WINAPI ActiveScriptSite_Release(IActiveScriptSite *iface)
661 {
662     return 1;
663 }
664
665 static HRESULT WINAPI ActiveScriptSite_GetLCID(IActiveScriptSite *iface, LCID *plcid)
666 {
667     *plcid = GetUserDefaultLCID();
668     return S_OK;
669 }
670
671 static HRESULT WINAPI ActiveScriptSite_GetItemInfo(IActiveScriptSite *iface, LPCOLESTR pstrName,
672         DWORD dwReturnMask, IUnknown **ppiunkItem, ITypeInfo **ppti)
673 {
674     ok(dwReturnMask == SCRIPTINFO_IUNKNOWN, "unexpected dwReturnMask %x\n", dwReturnMask);
675     ok(!ppti, "ppti != NULL\n");
676
677     if(!strcmp_wa(pstrName, test_valA))
678         CHECK_EXPECT(GetItemInfo_testVal);
679     else if(strcmp_wa(pstrName, testA))
680         ok(0, "unexpected pstrName %s\n", wine_dbgstr_w(pstrName));
681
682     *ppiunkItem = (IUnknown*)&Global;
683     return S_OK;
684 }
685
686 static HRESULT WINAPI ActiveScriptSite_GetDocVersionString(IActiveScriptSite *iface, BSTR *pbstrVersion)
687 {
688     return E_NOTIMPL;
689 }
690
691 static HRESULT WINAPI ActiveScriptSite_OnScriptTerminate(IActiveScriptSite *iface,
692         const VARIANT *pvarResult, const EXCEPINFO *pexcepinfo)
693 {
694     return E_NOTIMPL;
695 }
696
697 static HRESULT WINAPI ActiveScriptSite_OnStateChange(IActiveScriptSite *iface, SCRIPTSTATE ssScriptState)
698 {
699     return E_NOTIMPL;
700 }
701
702 static HRESULT WINAPI ActiveScriptSite_OnScriptError(IActiveScriptSite *iface, IActiveScriptError *pscripterror)
703 {
704     return E_NOTIMPL;
705 }
706
707 static HRESULT WINAPI ActiveScriptSite_OnEnterScript(IActiveScriptSite *iface)
708 {
709     return E_NOTIMPL;
710 }
711
712 static HRESULT WINAPI ActiveScriptSite_OnLeaveScript(IActiveScriptSite *iface)
713 {
714     return E_NOTIMPL;
715 }
716
717 #undef ACTSCPSITE_THIS
718
719 static const IActiveScriptSiteVtbl ActiveScriptSiteVtbl = {
720     ActiveScriptSite_QueryInterface,
721     ActiveScriptSite_AddRef,
722     ActiveScriptSite_Release,
723     ActiveScriptSite_GetLCID,
724     ActiveScriptSite_GetItemInfo,
725     ActiveScriptSite_GetDocVersionString,
726     ActiveScriptSite_OnScriptTerminate,
727     ActiveScriptSite_OnStateChange,
728     ActiveScriptSite_OnScriptError,
729     ActiveScriptSite_OnEnterScript,
730     ActiveScriptSite_OnLeaveScript
731 };
732
733 static IActiveScriptSite ActiveScriptSite = { &ActiveScriptSiteVtbl };
734
735 static HRESULT WINAPI ActiveScriptSite_OnScriptError_CheckError(IActiveScriptSite *iface, IActiveScriptError *pscripterror)
736 {
737     ok(pscripterror != NULL, "ActiveScriptSite_OnScriptError -- expected pscripterror to be set, got NULL\n");
738
739     script_error = pscripterror;
740     IUnknown_AddRef(script_error);
741
742     CHECK_EXPECT(ActiveScriptSite_OnScriptError);
743
744     return S_OK;
745 }
746
747 static const IActiveScriptSiteVtbl ActiveScriptSite_CheckErrorVtbl = {
748     ActiveScriptSite_QueryInterface,
749     ActiveScriptSite_AddRef,
750     ActiveScriptSite_Release,
751     ActiveScriptSite_GetLCID,
752     ActiveScriptSite_GetItemInfo,
753     ActiveScriptSite_GetDocVersionString,
754     ActiveScriptSite_OnScriptTerminate,
755     ActiveScriptSite_OnStateChange,
756     ActiveScriptSite_OnScriptError_CheckError,
757     ActiveScriptSite_OnEnterScript,
758     ActiveScriptSite_OnLeaveScript
759 };
760
761 static IActiveScriptSite ActiveScriptSite_CheckError = { &ActiveScriptSite_CheckErrorVtbl };
762
763 static HRESULT set_script_prop(IActiveScript *engine, DWORD property, VARIANT *val)
764 {
765     IActiveScriptProperty *script_prop;
766     HRESULT hres;
767
768     hres = IActiveScript_QueryInterface(engine, &IID_IActiveScriptProperty,
769             (void**)&script_prop);
770     ok(hres == S_OK, "Could not get IActiveScriptProperty iface: %08x\n", hres);
771
772     hres = IActiveScriptProperty_SetProperty(script_prop, property, NULL, val);
773     IActiveScriptProperty_Release(script_prop);
774
775     return hres;
776 }
777
778 static IActiveScript *create_script(void)
779 {
780     IActiveScript *script;
781     VARIANT v;
782     HRESULT hres;
783
784     hres = CoCreateInstance(&CLSID_JScript, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
785             &IID_IActiveScript, (void**)&script);
786     ok(hres == S_OK, "CoCreateInstance failed: %08x\n", hres);
787
788     V_VT(&v) = VT_I4;
789     V_I4(&v) = invoke_version;
790     hres = set_script_prop(script, SCRIPTPROP_INVOKEVERSIONING, &v);
791     ok(hres == S_OK || broken(hres == E_NOTIMPL), "SetProperty(SCRIPTPROP_INVOKEVERSIONING) failed: %08x\n", hres);
792     if(invoke_version && FAILED(hres)) {
793         IActiveScript_Release(script);
794         return NULL;
795     }
796
797     return script;
798 }
799
800 static HRESULT parse_script(DWORD flags, BSTR script_str)
801 {
802     IActiveScriptParse *parser;
803     IActiveScript *engine;
804     HRESULT hres;
805
806     engine = create_script();
807     if(!engine)
808         return S_OK;
809
810     hres = IActiveScript_QueryInterface(engine, &IID_IActiveScriptParse, (void**)&parser);
811     ok(hres == S_OK, "Could not get IActiveScriptParse: %08x\n", hres);
812     if (FAILED(hres))
813     {
814         IActiveScript_Release(engine);
815         return hres;
816     }
817
818     hres = IActiveScriptParse64_InitNew(parser);
819     ok(hres == S_OK, "InitNew failed: %08x\n", hres);
820
821     hres = IActiveScript_SetScriptSite(engine, &ActiveScriptSite);
822     ok(hres == S_OK, "SetScriptSite failed: %08x\n", hres);
823
824     hres = IActiveScript_AddNamedItem(engine, testW,
825             SCRIPTITEM_ISVISIBLE|SCRIPTITEM_ISSOURCE|flags);
826     ok(hres == S_OK, "AddNamedItem failed: %08x\n", hres);
827
828     hres = IActiveScript_SetScriptState(engine, SCRIPTSTATE_STARTED);
829     ok(hres == S_OK, "SetScriptState(SCRIPTSTATE_STARTED) failed: %08x\n", hres);
830
831     hres = IActiveScript_GetScriptDispatch(engine, NULL, &script_disp);
832     ok(hres == S_OK, "GetScriptDispatch failed: %08x\n", hres);
833     ok(script_disp != NULL, "script_disp == NULL\n");
834     ok(script_disp != (IDispatch*)&Global, "script_disp == Global\n");
835
836     hres = IActiveScriptParse64_ParseScriptText(parser, script_str, NULL, NULL, NULL, 0, 0, 0, NULL, NULL);
837
838     IDispatch_Release(script_disp);
839     IActiveScript_Release(engine);
840     IUnknown_Release(parser);
841
842     return hres;
843 }
844
845 static HRESULT parse_htmlscript(BSTR script_str)
846 {
847     IActiveScriptParse *parser;
848     IActiveScript *engine;
849     HRESULT hres;
850     BSTR tmp = a2bstr("</SCRIPT>");
851
852     engine = create_script();
853     if(!engine)
854         return E_FAIL;
855
856     hres = IActiveScript_QueryInterface(engine, &IID_IActiveScriptParse, (void**)&parser);
857     ok(hres == S_OK, "Could not get IActiveScriptParse: %08x\n", hres);
858     if (FAILED(hres))
859     {
860         IActiveScript_Release(engine);
861         return E_FAIL;
862     }
863
864     hres = IActiveScriptParse64_InitNew(parser);
865     ok(hres == S_OK, "InitNew failed: %08x\n", hres);
866
867     hres = IActiveScript_SetScriptSite(engine, &ActiveScriptSite);
868     ok(hres == S_OK, "SetScriptSite failed: %08x\n", hres);
869
870     hres = IActiveScript_AddNamedItem(engine, testW,
871             SCRIPTITEM_ISVISIBLE|SCRIPTITEM_ISSOURCE|SCRIPTITEM_GLOBALMEMBERS);
872     ok(hres == S_OK, "AddNamedItem failed: %08x\n", hres);
873
874     hres = IActiveScript_SetScriptState(engine, SCRIPTSTATE_STARTED);
875     ok(hres == S_OK, "SetScriptState(SCRIPTSTATE_STARTED) failed: %08x\n", hres);
876
877     hres = IActiveScriptParse64_ParseScriptText(parser, script_str, NULL, NULL, tmp, 0, 0, 0, NULL, NULL);
878
879     IActiveScript_Release(engine);
880     IUnknown_Release(parser);
881     SysFreeString(tmp);
882
883     return hres;
884 }
885
886 static void test_IActiveScriptError(IActiveScriptError *error, SCODE errorcode, ULONG line, LONG pos, BSTR script_source, BSTR description, BSTR line_text)
887 {
888     HRESULT hres;
889     DWORD source_context;
890     ULONG line_number;
891     LONG char_position;
892     BSTR linetext;
893     EXCEPINFO excep;
894
895     /* IActiveScriptError_GetSourcePosition */
896
897     hres = IActiveScriptError_GetSourcePosition(error, NULL, NULL, NULL);
898     ok(hres == S_OK, "IActiveScriptError_GetSourcePosition -- hres: expected S_OK, got 0x%08x\n", hres);
899
900     source_context = 0xdeadbeef;
901     hres = IActiveScriptError_GetSourcePosition(error, &source_context, NULL, NULL);
902     ok(hres == S_OK, "IActiveScriptError_GetSourcePosition -- hres: expected S_OK, got 0x%08x\n", hres);
903     ok(source_context == 0, "IActiveScriptError_GetSourcePosition -- source_context: expected 0, got 0x%08x\n", source_context);
904
905     line_number = 0xdeadbeef;
906     hres = IActiveScriptError_GetSourcePosition(error, NULL, &line_number, NULL);
907     ok(hres == S_OK, "IActiveScriptError_GetSourcePosition -- hres: expected S_OK, got 0x%08x\n", hres);
908     ok(line_number == line, "IActiveScriptError_GetSourcePosition -- line_number: expected %d, got %d\n", line, line_number);
909
910     char_position = 0xdeadbeef;
911     hres = IActiveScriptError_GetSourcePosition(error, NULL, NULL, &char_position);
912     ok(hres == S_OK, "IActiveScriptError_GetSourcePosition -- hres: expected S_OK, got 0x%08x\n", hres);
913     ok(char_position == pos, "IActiveScriptError_GetSourcePosition -- char_position: expected %d, got %d\n", pos, char_position);
914
915     /* IActiveScriptError_GetSourceLineText */
916
917     hres = IActiveScriptError_GetSourceLineText(error, NULL);
918     ok(hres == E_POINTER, "IActiveScriptError_GetSourceLineText -- hres: expected E_POINTER, got 0x%08x\n", hres);
919
920     linetext = NULL;
921     hres = IActiveScriptError_GetSourceLineText(error, &linetext);
922     if (line_text) {
923         ok(hres == S_OK, "IActiveScriptError_GetSourceLineText -- hres: expected S_OK, got 0x%08x\n", hres);
924         ok(linetext != NULL && !lstrcmpW(linetext, line_text),
925            "IActiveScriptError_GetSourceLineText -- expected %s, got %s\n", wine_dbgstr_w(line_text), wine_dbgstr_w(linetext));
926     } else {
927         ok(hres == E_FAIL, "IActiveScriptError_GetSourceLineText -- hres: expected S_OK, got 0x%08x\n", hres);
928         ok(linetext == NULL,
929            "IActiveScriptError_GetSourceLineText -- expected NULL, got %s\n", wine_dbgstr_w(linetext));
930     }
931     SysFreeString(linetext);
932
933     /* IActiveScriptError_GetExceptionInfo */
934
935     hres = IActiveScriptError_GetExceptionInfo(error, NULL);
936     ok(hres == E_POINTER, "IActiveScriptError_GetExceptionInfo -- hres: expected E_POINTER, got 0x%08x\n", hres);
937
938     excep.wCode = 0xdead;
939     excep.wReserved = 0xdead;
940     excep.bstrSource = (BSTR)0xdeadbeef;
941     excep.bstrDescription = (BSTR)0xdeadbeef;
942     excep.bstrHelpFile = (BSTR)0xdeadbeef;
943     excep.dwHelpContext = 0xdeadbeef;
944     excep.pvReserved = (void *)0xdeadbeef;
945     excep.pfnDeferredFillIn = (void *)0xdeadbeef;
946     excep.scode = 0xdeadbeef;
947
948     hres = IActiveScriptError_GetExceptionInfo(error, &excep);
949     ok(hres == S_OK, "IActiveScriptError_GetExceptionInfo -- hres: expected S_OK, got 0x%08x\n", hres);
950
951     ok(excep.wCode == 0, "IActiveScriptError_GetExceptionInfo -- excep.wCode: expected 0, got 0x%08x\n", excep.wCode);
952     ok(excep.wReserved == 0, "IActiveScriptError_GetExceptionInfo -- excep.wReserved: expected 0, got %d\n", excep.wReserved);
953     if (PRIMARYLANGID(LANGIDFROMLCID(GetThreadLocale())) != LANG_ENGLISH)
954         skip("Non-english locale (test with hardcoded strings)\n");
955     else {
956         ok(excep.bstrSource != NULL && !lstrcmpW(excep.bstrSource, script_source),
957            "IActiveScriptError_GetExceptionInfo -- excep.bstrSource is not valid: expected %s, got %s\n",
958            wine_dbgstr_w(script_source), wine_dbgstr_w(excep.bstrSource));
959         ok(excep.bstrDescription != NULL && !lstrcmpW(excep.bstrDescription, description),
960            "IActiveScriptError_GetExceptionInfo -- excep.bstrDescription is not valid: got %s\n", wine_dbgstr_w(excep.bstrDescription));
961     }
962     ok(excep.bstrHelpFile == NULL,
963        "IActiveScriptError_GetExceptionInfo -- excep.bstrHelpFile: expected NULL, got %s\n", wine_dbgstr_w(excep.bstrHelpFile));
964     ok(excep.dwHelpContext == 0, "IActiveScriptError_GetExceptionInfo -- excep.dwHelpContext: expected 0, got %d\n", excep.dwHelpContext);
965     ok(excep.pvReserved == NULL, "IActiveScriptError_GetExceptionInfo -- excep.pvReserved: expected NULL, got %p\n", excep.pvReserved);
966     ok(excep.pfnDeferredFillIn == NULL, "IActiveScriptError_GetExceptionInfo -- excep.pfnDeferredFillIn: expected NULL, got %p\n", excep.pfnDeferredFillIn);
967     ok(excep.scode == errorcode, "IActiveScriptError_GetExceptionInfo -- excep.scode: expected 0x%08x, got 0x%08x\n", errorcode, excep.scode);
968
969     SysFreeString(excep.bstrSource);
970     SysFreeString(excep.bstrDescription);
971     SysFreeString(excep.bstrHelpFile);
972 }
973
974 static void parse_script_with_error(DWORD flags, BSTR script_str, SCODE errorcode, ULONG line, LONG pos, BSTR script_source, BSTR description, BSTR line_text)
975 {
976     IActiveScriptParse *parser;
977     IActiveScript *engine;
978     HRESULT hres;
979
980     engine = create_script();
981     if(!engine)
982         return;
983
984     hres = IActiveScript_QueryInterface(engine, &IID_IActiveScriptParse, (void**)&parser);
985     ok(hres == S_OK, "Could not get IActiveScriptParse: %08x\n", hres);
986     if (FAILED(hres))
987     {
988         IActiveScript_Release(engine);
989         return;
990     }
991
992     hres = IActiveScriptParse64_InitNew(parser);
993     ok(hres == S_OK, "InitNew failed: %08x\n", hres);
994
995     hres = IActiveScript_SetScriptSite(engine, &ActiveScriptSite_CheckError);
996     ok(hres == S_OK, "SetScriptSite failed: %08x\n", hres);
997
998     hres = IActiveScript_AddNamedItem(engine, testW,
999             SCRIPTITEM_ISVISIBLE|SCRIPTITEM_ISSOURCE|flags);
1000     ok(hres == S_OK, "AddNamedItem failed: %08x\n", hres);
1001
1002     hres = IActiveScript_SetScriptState(engine, SCRIPTSTATE_STARTED);
1003     ok(hres == S_OK, "SetScriptState(SCRIPTSTATE_STARTED) failed: %08x\n", hres);
1004
1005     hres = IActiveScript_GetScriptDispatch(engine, NULL, &script_disp);
1006     ok(hres == S_OK, "GetScriptDispatch failed: %08x\n", hres);
1007     ok(script_disp != NULL, "script_disp == NULL\n");
1008     ok(script_disp != (IDispatch*)&Global, "script_disp == Global\n");
1009
1010     script_error = NULL;
1011     SET_EXPECT(ActiveScriptSite_OnScriptError);
1012     hres = IActiveScriptParse64_ParseScriptText(parser, script_str, NULL, NULL, NULL, 0, 0, 0, NULL, NULL);
1013     todo_wine ok(hres == 0x80020101, "parse_script_with_error should have returned 0x80020101, got: 0x%08x\n", hres);
1014     todo_wine CHECK_CALLED(ActiveScriptSite_OnScriptError);
1015
1016     if (script_error)
1017     {
1018         test_IActiveScriptError(script_error, errorcode, line, pos, script_source, description, line_text);
1019
1020         IUnknown_Release(script_error);
1021     }
1022
1023     IDispatch_Release(script_disp);
1024     IActiveScript_Release(engine);
1025     IUnknown_Release(parser);
1026 }
1027
1028 static void parse_script_af(DWORD flags, const char *src)
1029 {
1030     BSTR tmp;
1031     HRESULT hres;
1032
1033     tmp = a2bstr(src);
1034     hres = parse_script(flags, tmp);
1035     SysFreeString(tmp);
1036     ok(hres == S_OK, "parse_script failed: %08x\n", hres);
1037 }
1038
1039 static void parse_script_a(const char *src)
1040 {
1041     parse_script_af(SCRIPTITEM_GLOBALMEMBERS, src);
1042 }
1043
1044 static void parse_script_with_error_a(const char *src, SCODE errorcode, ULONG line, LONG pos, LPCSTR source, LPCSTR desc, LPCSTR linetext)
1045 {
1046     BSTR tmp, script_source, description, line_text;
1047
1048     tmp = a2bstr(src);
1049     script_source = a2bstr(source);
1050     description = a2bstr(desc);
1051     line_text = a2bstr(linetext);
1052
1053     parse_script_with_error(SCRIPTITEM_GLOBALMEMBERS, tmp, errorcode, line, pos, script_source, description, line_text);
1054
1055     SysFreeString(line_text);
1056     SysFreeString(description);
1057     SysFreeString(script_source);
1058     SysFreeString(tmp);
1059 }
1060
1061 static HRESULT parse_htmlscript_a(const char *src)
1062 {
1063     HRESULT hres;
1064     BSTR tmp = a2bstr(src);
1065     hres = parse_htmlscript(tmp);
1066     SysFreeString(tmp);
1067
1068     return hres;
1069 }
1070
1071 static BSTR get_script_from_file(const char *filename)
1072 {
1073     DWORD size, len;
1074     HANDLE file, map;
1075     const char *file_map;
1076     BSTR ret;
1077
1078     file = CreateFileA(filename, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_READONLY, NULL);
1079     if(file == INVALID_HANDLE_VALUE) {
1080         trace("Could not open file: %u\n", GetLastError());
1081         return NULL;
1082     }
1083
1084     size = GetFileSize(file, NULL);
1085
1086     map = CreateFileMapping(file, NULL, PAGE_READONLY, 0, 0, NULL);
1087     CloseHandle(file);
1088     if(map == INVALID_HANDLE_VALUE) {
1089         trace("Could not create file mapping: %u\n", GetLastError());
1090         return NULL;
1091     }
1092
1093     file_map = MapViewOfFile(map, FILE_MAP_READ, 0, 0, 0);
1094     CloseHandle(map);
1095     if(!file_map) {
1096         trace("MapViewOfFile failed: %u\n", GetLastError());
1097         return NULL;
1098     }
1099
1100     len = MultiByteToWideChar(CP_ACP, 0, file_map, size, NULL, 0);
1101     ret = SysAllocStringLen(NULL, len);
1102     MultiByteToWideChar(CP_ACP, 0, file_map, size, ret, len);
1103
1104     UnmapViewOfFile(file_map);
1105
1106     return ret;
1107 }
1108
1109 static void run_from_file(const char *filename)
1110 {
1111     BSTR script_str;
1112     HRESULT hres;
1113
1114     script_str = get_script_from_file(filename);
1115     if(!script_str)
1116         return;
1117
1118     strict_dispid_check = FALSE;
1119     hres = parse_script(SCRIPTITEM_GLOBALMEMBERS, script_str);
1120     SysFreeString(script_str);
1121     ok(hres == S_OK, "parse_script failed: %08x\n", hres);
1122 }
1123
1124 static void run_from_res(const char *name)
1125 {
1126     const char *data;
1127     DWORD size, len;
1128     BSTR str;
1129     HRSRC src;
1130     HRESULT hres;
1131
1132     strict_dispid_check = FALSE;
1133     test_name = name;
1134
1135     src = FindResourceA(NULL, name, (LPCSTR)40);
1136     ok(src != NULL, "Could not find resource %s\n", name);
1137
1138     size = SizeofResource(NULL, src);
1139     data = LoadResource(NULL, src);
1140
1141     len = MultiByteToWideChar(CP_ACP, 0, data, size, NULL, 0);
1142     str = SysAllocStringLen(NULL, len);
1143     len = MultiByteToWideChar(CP_ACP, 0, data, size, str, len);
1144
1145     SET_EXPECT(global_success_d);
1146     SET_EXPECT(global_success_i);
1147     hres = parse_script(SCRIPTITEM_GLOBALMEMBERS, str);
1148     CHECK_CALLED(global_success_d);
1149     CHECK_CALLED(global_success_i);
1150
1151     ok(hres == S_OK, "parse_script failed: %08x\n", hres);
1152     SysFreeString(str);
1153 }
1154
1155 static void test_isvisible(BOOL global_members)
1156 {
1157     IActiveScriptParse *parser;
1158     IActiveScript *engine;
1159     HRESULT hres;
1160
1161     static const WCHAR script_textW[] =
1162         {'v','a','r',' ','v',' ','=',' ','t','e','s','t','V','a','l',';',0};
1163
1164     engine = create_script();
1165     if(!engine)
1166         return;
1167
1168     hres = IActiveScript_QueryInterface(engine, &IID_IActiveScriptParse, (void**)&parser);
1169     ok(hres == S_OK, "Could not get IActiveScriptParse: %08x\n", hres);
1170     if (FAILED(hres))
1171     {
1172         IActiveScript_Release(engine);
1173         return;
1174     }
1175
1176     hres = IActiveScriptParse64_InitNew(parser);
1177     ok(hres == S_OK, "InitNew failed: %08x\n", hres);
1178
1179     hres = IActiveScript_SetScriptSite(engine, &ActiveScriptSite);
1180     ok(hres == S_OK, "SetScriptSite failed: %08x\n", hres);
1181
1182     if(global_members)
1183         SET_EXPECT(GetItemInfo_testVal);
1184     hres = IActiveScript_AddNamedItem(engine, test_valW,
1185             SCRIPTITEM_ISVISIBLE|SCRIPTITEM_ISSOURCE|
1186             (global_members ? SCRIPTITEM_GLOBALMEMBERS : 0));
1187     ok(hres == S_OK, "AddNamedItem failed: %08x\n", hres);
1188     if(global_members)
1189         CHECK_CALLED(GetItemInfo_testVal);
1190
1191     hres = IActiveScript_SetScriptState(engine, SCRIPTSTATE_STARTED);
1192     ok(hres == S_OK, "SetScriptState(SCRIPTSTATE_STARTED) failed: %08x\n", hres);
1193
1194     if(!global_members)
1195         SET_EXPECT(GetItemInfo_testVal);
1196     hres = IActiveScriptParse64_ParseScriptText(parser, script_textW, NULL, NULL, NULL, 0, 0, 0, NULL, NULL);
1197     ok(hres == S_OK, "ParseScriptText failed: %08x\n", hres);
1198     if(!global_members)
1199         CHECK_CALLED(GetItemInfo_testVal);
1200
1201     hres = IActiveScriptParse64_ParseScriptText(parser, script_textW, NULL, NULL, NULL, 0, 0, 0, NULL, NULL);
1202     ok(hres == S_OK, "ParseScriptText failed: %08x\n", hres);
1203
1204     IActiveScript_Release(engine);
1205     IUnknown_Release(parser);
1206 }
1207
1208 static void run_tests(void)
1209 {
1210     HRESULT hres;
1211
1212     if(invoke_version) {
1213         IActiveScript *script;
1214
1215         script = create_script();
1216         if(!script) {
1217             win_skip("Could not create script\n");
1218             return;
1219         }
1220         IActiveScript_Release(script);
1221     }
1222
1223     strict_dispid_check = TRUE;
1224
1225     parse_script_a("");
1226     parse_script_a("/* empty */ ;");
1227
1228     SET_EXPECT(global_propget_d);
1229     SET_EXPECT(global_propget_i);
1230     parse_script_a("testPropGet;");
1231     CHECK_CALLED(global_propget_d);
1232     CHECK_CALLED(global_propget_i);
1233
1234     SET_EXPECT(global_propput_d);
1235     SET_EXPECT(global_propput_i);
1236     parse_script_a("testPropPut = 1;");
1237     CHECK_CALLED(global_propput_d);
1238     CHECK_CALLED(global_propput_i);
1239
1240     SET_EXPECT(global_success_d);
1241     SET_EXPECT(global_success_i);
1242     parse_script_a("reportSuccess();");
1243     CHECK_CALLED(global_success_d);
1244     CHECK_CALLED(global_success_i);
1245
1246     SET_EXPECT(testobj_delete);
1247     parse_script_a("delete testObj.deleteTest;");
1248     CHECK_CALLED(testobj_delete);
1249
1250     parse_script_a("ok(typeof(test) === 'object', \"typeof(test) != 'object'\");");
1251
1252     parse_script_a("function reportSuccess() {}; reportSuccess();");
1253
1254     SET_EXPECT(global_propget_d);
1255     parse_script_a("var testPropGet");
1256     CHECK_CALLED(global_propget_d);
1257
1258     SET_EXPECT(global_propget_d);
1259     parse_script_a("eval('var testPropGet;');");
1260     CHECK_CALLED(global_propget_d);
1261
1262     SET_EXPECT(global_notexists_d);
1263     parse_script_a("var notExists; notExists = 1;");
1264     CHECK_CALLED(global_notexists_d);
1265
1266     parse_script_a("function f() { var testPropGet; }");
1267     parse_script_a("(function () { var testPropGet; })();");
1268     parse_script_a("(function () { eval('var testPropGet;'); })();");
1269
1270     parse_script_a("ok((testObj instanceof Object) === false, 'testObj is instance of Object');");
1271
1272     SET_EXPECT(testobj_prop_d);
1273     parse_script_a("ok(('prop' in testObj) === true, 'prop is not in testObj');");
1274     CHECK_CALLED(testobj_prop_d);
1275
1276     SET_EXPECT(testobj_noprop_d);
1277     parse_script_a("ok(('noprop' in testObj) === false, 'noprop is in testObj');");
1278     CHECK_CALLED(testobj_noprop_d);
1279
1280     SET_EXPECT(testobj_value);
1281     parse_script_a("ok(String(testObj) === '1', 'wrong testObj value');");
1282     CHECK_CALLED(testobj_value);
1283
1284     SET_EXPECT(testobj_value);
1285     parse_script_a("ok(String.prototype.concat.call(testObj, ' OK') === '1 OK', 'wrong concat result');");
1286     CHECK_CALLED(testobj_value);
1287
1288     SET_EXPECT(global_propget_d);
1289     SET_EXPECT(global_propget_i);
1290     parse_script_a("this.testPropGet;");
1291     CHECK_CALLED(global_propget_d);
1292     CHECK_CALLED(global_propget_i);
1293
1294     SET_EXPECT(global_propget_d);
1295     SET_EXPECT(global_propget_i);
1296     parse_script_a("(function () { this.testPropGet; })();");
1297     CHECK_CALLED(global_propget_d);
1298     CHECK_CALLED(global_propget_i);
1299
1300     parse_script_a("testThis(this);");
1301     parse_script_a("(function () { testThis(this); })();");
1302
1303     run_from_res("lang.js");
1304     run_from_res("api.js");
1305     run_from_res("regexp.js");
1306
1307     test_isvisible(FALSE);
1308     test_isvisible(TRUE);
1309
1310     parse_script_af(0, "test.testThis2(this);");
1311     parse_script_af(0, "(function () { test.testThis2(this); })();");
1312
1313     hres = parse_htmlscript_a("<!--");
1314     ok(hres == S_OK, "ParseScriptText failed: %08x\n", hres);
1315     hres = parse_htmlscript_a("-->");
1316     ok(hres == S_OK, "ParseScriptText failed: %08x\n", hres);
1317     hres = parse_htmlscript_a("<!--\nvar a=1;\n-->\n");
1318     ok(hres == S_OK, "ParseScriptText failed: %08x\n", hres);
1319     hres = parse_htmlscript_a("<!--\n<!-- ignore this\n-->\n");
1320     ok(hres == S_OK, "ParseScriptText failed: %08x\n", hres);
1321     hres = parse_htmlscript_a("var a=1;\nif(a-->0) a=5;\n");
1322     ok(hres == S_OK, "ParseScriptText failed: %08x\n", hres);
1323     hres = parse_htmlscript_a("var a=1;\nif(a\n-->0) a=5;\n");
1324     ok(hres != S_OK, "ParseScriptText have not failed\n");
1325
1326     parse_script_with_error_a(
1327         "?",
1328         0x800a03ea, 0, 0,
1329         "Microsoft JScript compilation error",
1330         "Syntax error",
1331         "?");
1332
1333     parse_script_with_error_a(
1334         "var a=1;\nif(a\n-->0) a=5;\n",
1335         0x800a03ee, 2, 0,
1336         "Microsoft JScript compilation error",
1337         "Expected ')'",
1338         "-->0) a=5;");
1339
1340     parse_script_with_error_a(
1341         "new 3;",
1342         0x800a01bd, 0, 0,
1343         "Microsoft JScript runtime error",
1344         "Object doesn't support this action",
1345         NULL);
1346
1347     parse_script_with_error_a(
1348         "new null;",
1349         0x800a138f, 0, 0,
1350         "Microsoft JScript runtime error",
1351         "Object expected",
1352         NULL);
1353
1354     parse_script_with_error_a(
1355         "var a;\nnew null;",
1356         0x800a138f, 1, 0,
1357         "Microsoft JScript runtime error",
1358         "Object expected",
1359         NULL);
1360
1361     parse_script_with_error_a(
1362         "var a; new null;",
1363         0x800a138f, 0, 7,
1364         "Microsoft JScript runtime error",
1365         "Object expected",
1366         NULL);
1367 }
1368
1369 static BOOL check_jscript(void)
1370 {
1371     IActiveScriptProperty *script_prop;
1372     BSTR str;
1373     HRESULT hres;
1374
1375     hres = CoCreateInstance(&CLSID_JScript, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
1376             &IID_IActiveScriptProperty, (void**)&script_prop);
1377     if(FAILED(hres))
1378         return FALSE;
1379     IActiveScriptProperty_Release(script_prop);
1380
1381     str = a2bstr("if(!('localeCompare' in String.prototype)) throw 1;");
1382     hres = parse_script(0, str);
1383     SysFreeString(str);
1384
1385     return hres == S_OK;
1386 }
1387
1388 START_TEST(run)
1389 {
1390     int argc;
1391     char **argv;
1392
1393     argc = winetest_get_mainargs(&argv);
1394
1395     CoInitialize(NULL);
1396
1397     if(!check_jscript()) {
1398         win_skip("Broken engine, probably too old\n");
1399     }else if(argc > 2) {
1400         run_from_file(argv[2]);
1401     }else {
1402         trace("invoke version 0\n");
1403         invoke_version = 0;
1404         run_tests();
1405
1406         trace("invoke version 2\n");
1407         invoke_version = 2;
1408         run_tests();
1409     }
1410
1411     CoUninitialize();
1412 }