1 /*
2  * Unit tests for OLE storage
3  *
4  * Copyright (c) 2004 Mike McCormack
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  */
20 
21 #include <stdio.h>
22 
23 #define COBJMACROS
24 #ifdef __REACTOS__
25 #define CONST_VTABLE
26 #else
27 #define NONAMELESSUNION
28 #define NONAMELESSSTRUCT
29 #endif
30 
31 #include <windows.h>
32 #include "wine/test.h"
33 
34 #include "ole2.h"
35 #include "objidl.h"
36 #include "initguid.h"
37 
38 DEFINE_GUID( test_stg_cls, 0x88888888, 0x0425, 0x0000, 0,0,0,0,0,0,0,0);
39 
40 #define ok_ole_success(hr, func) ok(hr == S_OK, func " failed with error 0x%08x\n", hr)
41 
42 static CHAR filenameA[MAX_PATH];
43 static WCHAR filename[MAX_PATH];
44 
45 static const char file1_nameA[] = {'c','o','p','y','t','e','s','t','A',0};
46 static const WCHAR file1_name[] = {'c','o','p','y','t','e','s','t','A',0};
47 static const char file2_nameA[] = {'c','o','p','y','t','e','s','t','B',0};
48 static const WCHAR file2_name[] = {'c','o','p','y','t','e','s','t','B',0};
49 static const WCHAR stgA_name[] = {'S','t','o','r','a','g','e','A',0};
50 static const WCHAR stgB_name[] = {'S','t','o','r','a','g','e','B',0};
51 static const WCHAR strmA_name[] = {'S','t','r','e','a','m','A',0};
52 static const WCHAR strmB_name[] = {'S','t','r','e','a','m','B',0};
53 static const WCHAR strmC_name[] = {'S','t','r','e','a','m','C',0};
54 
55 /* Win9x and WinMe don't have lstrcmpW */
strcmp_ww(LPCWSTR strw1,LPCWSTR strw2)56 static int strcmp_ww(LPCWSTR strw1, LPCWSTR strw2)
57 {
58     CHAR stra1[512], stra2[512];
59     WideCharToMultiByte(CP_ACP, 0, strw1, -1, stra1, sizeof(stra1), NULL, NULL);
60     WideCharToMultiByte(CP_ACP, 0, strw2, -1, stra2, sizeof(stra2), NULL, NULL);
61     return lstrcmpA(stra1, stra2);
62 }
63 
64 typedef struct TestLockBytes {
65     ILockBytes ILockBytes_iface;
66     LONG ref;
67     BYTE* contents;
68     ULONG size;
69     ULONG buffer_size;
70     HRESULT lock_hr;
71     ULONG locks_supported;
72     ULONG lock_called;
73 } TestLockBytes;
74 
impl_from_ILockBytes(ILockBytes * iface)75 static inline TestLockBytes *impl_from_ILockBytes(ILockBytes *iface)
76 {
77     return CONTAINING_RECORD(iface, TestLockBytes, ILockBytes_iface);
78 }
79 
TestLockBytes_QueryInterface(ILockBytes * iface,REFIID iid,void ** ppv)80 static HRESULT WINAPI TestLockBytes_QueryInterface(ILockBytes *iface, REFIID iid,
81     void **ppv)
82 {
83     TestLockBytes *This = impl_from_ILockBytes(iface);
84 
85     if (!ppv) return E_INVALIDARG;
86 
87     if (IsEqualIID(&IID_IUnknown, iid) ||
88         IsEqualIID(&IID_ILockBytes, iid))
89         *ppv = &This->ILockBytes_iface;
90     else
91         return E_NOINTERFACE;
92 
93     IUnknown_AddRef((IUnknown*)*ppv);
94     return S_OK;
95 }
96 
TestLockBytes_AddRef(ILockBytes * iface)97 static ULONG WINAPI TestLockBytes_AddRef(ILockBytes *iface)
98 {
99     TestLockBytes *This = impl_from_ILockBytes(iface);
100     ULONG ref = InterlockedIncrement(&This->ref);
101     return ref;
102 }
103 
TestLockBytes_Release(ILockBytes * iface)104 static ULONG WINAPI TestLockBytes_Release(ILockBytes *iface)
105 {
106     TestLockBytes *This = impl_from_ILockBytes(iface);
107     ULONG ref = InterlockedDecrement(&This->ref);
108     return ref;
109 }
110 
TestLockBytes_ReadAt(ILockBytes * iface,ULARGE_INTEGER ulOffset,void * pv,ULONG cb,ULONG * pcbRead)111 static HRESULT WINAPI TestLockBytes_ReadAt(ILockBytes *iface,
112     ULARGE_INTEGER ulOffset, void *pv, ULONG cb, ULONG *pcbRead)
113 {
114     TestLockBytes *This = impl_from_ILockBytes(iface);
115     ULONG dummy;
116 
117     if (!pv) return E_INVALIDARG;
118 
119     if (!pcbRead) pcbRead = &dummy;
120 
121     if (ulOffset.QuadPart >= This->size)
122     {
123         *pcbRead = 0;
124         return S_OK;
125     }
126 
127     cb = min(cb, This->size - ulOffset.QuadPart);
128 
129     *pcbRead = cb;
130     memcpy(pv, &This->contents[ulOffset.QuadPart], cb);
131 
132     return S_OK;
133 }
134 
TestLockBytes_WriteAt(ILockBytes * iface,ULARGE_INTEGER ulOffset,const void * pv,ULONG cb,ULONG * pcbWritten)135 static HRESULT WINAPI TestLockBytes_WriteAt(ILockBytes *iface,
136     ULARGE_INTEGER ulOffset, const void *pv, ULONG cb, ULONG *pcbWritten)
137 {
138     TestLockBytes *This = impl_from_ILockBytes(iface);
139     HRESULT hr;
140     ULONG dummy;
141 
142     if (!pv) return E_INVALIDARG;
143 
144     if (!pcbWritten) pcbWritten = &dummy;
145 
146     if (ulOffset.QuadPart + cb > This->size)
147     {
148         ULARGE_INTEGER new_size;
149         new_size.QuadPart = ulOffset.QuadPart + cb;
150         hr = ILockBytes_SetSize(iface, new_size);
151         if (FAILED(hr)) return hr;
152     }
153 
154     *pcbWritten = cb;
155     memcpy(&This->contents[ulOffset.QuadPart], pv, cb);
156 
157     return S_OK;
158 }
159 
TestLockBytes_Flush(ILockBytes * iface)160 static HRESULT WINAPI TestLockBytes_Flush(ILockBytes *iface)
161 {
162     return S_OK;
163 }
164 
TestLockBytes_SetSize(ILockBytes * iface,ULARGE_INTEGER cb)165 static HRESULT WINAPI TestLockBytes_SetSize(ILockBytes *iface,
166     ULARGE_INTEGER cb)
167 {
168     TestLockBytes *This = impl_from_ILockBytes(iface);
169 
170     if (This->buffer_size < cb.QuadPart)
171     {
172         ULONG new_buffer_size = max(This->buffer_size * 2, cb.QuadPart);
173         BYTE* new_buffer = HeapAlloc(GetProcessHeap(), 0, new_buffer_size);
174         if (!new_buffer) return E_OUTOFMEMORY;
175         memcpy(new_buffer, This->contents, This->size);
176         HeapFree(GetProcessHeap(), 0, This->contents);
177         This->contents = new_buffer;
178     }
179 
180     if (cb.QuadPart > This->size)
181         memset(&This->contents[This->size], 0, cb.QuadPart - This->size);
182 
183     This->size = cb.QuadPart;
184 
185     return S_OK;
186 }
187 
TestLockBytes_LockRegion(ILockBytes * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)188 static HRESULT WINAPI TestLockBytes_LockRegion(ILockBytes *iface,
189     ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
190 {
191     TestLockBytes *This = impl_from_ILockBytes(iface);
192     This->lock_called++;
193     return This->lock_hr;
194 }
195 
TestLockBytes_UnlockRegion(ILockBytes * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)196 static HRESULT WINAPI TestLockBytes_UnlockRegion(ILockBytes *iface,
197     ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
198 {
199     TestLockBytes *This = impl_from_ILockBytes(iface);
200     return This->lock_hr;
201 }
202 
TestLockBytes_Stat(ILockBytes * iface,STATSTG * pstatstg,DWORD grfStatFlag)203 static HRESULT WINAPI TestLockBytes_Stat(ILockBytes *iface,
204     STATSTG *pstatstg, DWORD grfStatFlag)
205 {
206     TestLockBytes *This = impl_from_ILockBytes(iface);
207     static const WCHAR dummy_name[] = {'d','u','m','m','y',0};
208 
209     if (!pstatstg) return E_INVALIDARG;
210 
211     memset(pstatstg, 0, sizeof(STATSTG));
212 
213     if (!(grfStatFlag & STATFLAG_NONAME))
214     {
215         pstatstg->pwcsName = CoTaskMemAlloc(sizeof(dummy_name));
216         if (!pstatstg->pwcsName) return E_OUTOFMEMORY;
217         memcpy(pstatstg->pwcsName, dummy_name, sizeof(dummy_name));
218     }
219 
220     pstatstg->type = STGTY_LOCKBYTES;
221     pstatstg->cbSize.QuadPart = This->size;
222     pstatstg->grfLocksSupported = This->locks_supported;
223 
224     return S_OK;
225 }
226 
227 static const ILockBytesVtbl TestLockBytes_Vtbl = {
228     TestLockBytes_QueryInterface,
229     TestLockBytes_AddRef,
230     TestLockBytes_Release,
231     TestLockBytes_ReadAt,
232     TestLockBytes_WriteAt,
233     TestLockBytes_Flush,
234     TestLockBytes_SetSize,
235     TestLockBytes_LockRegion,
236     TestLockBytes_UnlockRegion,
237     TestLockBytes_Stat
238 };
239 
CreateTestLockBytes(TestLockBytes ** This)240 static void CreateTestLockBytes(TestLockBytes **This)
241 {
242     *This = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(**This));
243 
244     if (*This)
245     {
246         (*This)->ILockBytes_iface.lpVtbl = &TestLockBytes_Vtbl;
247         (*This)->ref = 1;
248     }
249 }
250 
DeleteTestLockBytes(TestLockBytes * This)251 static void DeleteTestLockBytes(TestLockBytes *This)
252 {
253     ok(This->ILockBytes_iface.lpVtbl == &TestLockBytes_Vtbl, "test lock bytes %p deleted with incorrect vtable\n", This);
254     ok(This->ref == 1, "test lock bytes %p deleted with %i references instead of 1\n", This, This->ref);
255     HeapFree(GetProcessHeap(), 0, This->contents);
256     HeapFree(GetProcessHeap(), 0, This);
257 }
258 
test_hglobal_storage_stat(void)259 static void test_hglobal_storage_stat(void)
260 {
261     ILockBytes *ilb = NULL;
262     IStorage *stg = NULL;
263     HRESULT r;
264     STATSTG stat;
265     DWORD mode, refcount;
266 
267     r = CreateILockBytesOnHGlobal( NULL, TRUE, &ilb );
268     ok( r == S_OK, "CreateILockBytesOnHGlobal failed\n");
269 
270     r = StgIsStorageILockBytes( ilb );
271     ok( r == S_FALSE, "StgIsStorageILockBytes should have failed\n");
272 
273     mode = STGM_CREATE|STGM_SHARE_EXCLUSIVE|STGM_READWRITE;/*0x1012*/
274     r = StgCreateDocfileOnILockBytes( ilb, mode, 0,  &stg );
275     ok( r == S_OK, "StgCreateDocfileOnILockBytes failed\n");
276 
277     r = WriteClassStg( stg, &test_stg_cls );
278     ok( r == S_OK, "WriteClassStg failed\n");
279 
280     r = StgIsStorageILockBytes( ilb );
281     ok( r == S_OK, "StgIsStorageILockBytes failed\n");
282 
283     memset( &stat, 0, sizeof stat );
284     r = IStorage_Stat( stg, &stat, 0 );
285 
286     ok( stat.pwcsName == NULL, "storage name not null\n");
287     ok( stat.type == 1, "type is wrong\n");
288     ok( stat.grfMode == 0x12, "grf mode is incorrect\n");
289     ok( !memcmp(&stat.clsid, &test_stg_cls, sizeof test_stg_cls), "CLSID is wrong\n");
290 
291     refcount = IStorage_Release( stg );
292     ok( refcount == 0, "IStorage refcount is wrong\n");
293     refcount = ILockBytes_Release( ilb );
294     ok( refcount == 0, "ILockBytes refcount is wrong\n");
295 }
296 
test_create_storage_modes(void)297 static void test_create_storage_modes(void)
298 {
299    IStorage *stg = NULL;
300    HRESULT r;
301 
302    DeleteFileA(filenameA);
303 
304    /* test with some invalid parameters */
305    r = StgCreateDocfile( NULL, 0, 0, &stg);
306    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
307    r = StgCreateDocfile( filename, 0, 0, &stg);
308    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
309    r = StgCreateDocfile( filename, STGM_CREATE, 0, &stg);
310    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
311    r = StgCreateDocfile( filename, STGM_CREATE | STGM_READWRITE, 0, &stg);
312    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
313    r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE, 0, &stg);
314    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
315    r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, NULL);
316    ok(r==STG_E_INVALIDPOINTER, "StgCreateDocfile succeeded\n");
317    r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 1, &stg);
318    ok(r==STG_E_INVALIDPARAMETER, "StgCreateDocfile succeeded\n");
319    r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_DENY_WRITE | STGM_READWRITE, 0, &stg);
320    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
321    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READ, 0, &stg);
322    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
323    r = StgCreateDocfile( filename, STGM_PRIORITY, 0, &stg);
324    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
325 
326    /* StgCreateDocfile seems to be very particular about the flags it accepts */
327    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED | STGM_WRITE, 0, &stg);
328    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
329    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED | 8, 0, &stg);
330    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
331    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED | 0x80, 0, &stg);
332    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
333    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED | 0x800, 0, &stg);
334    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
335    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED | 0x8000, 0, &stg);
336    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
337    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED | 0x80000, 0, &stg);
338    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
339    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED | 0x800000, 0, &stg);
340    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
341    ok(stg == NULL, "stg was set\n");
342 
343    /* check what happens if the file already exists (which is how it's meant to be used) */
344    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stg);
345    ok(r==S_OK, "StgCreateDocfile failed\n");
346    r = IStorage_Release(stg);
347    ok(r == 0, "storage not released\n");
348    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
349    ok(r==STG_E_FILEALREADYEXISTS, "StgCreateDocfile wrong error\n"); /* FAILIFTHERE is default */
350    r = StgCreateDocfile( filename, STGM_READ, 0, &stg);
351    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n"); /* need at least readmode and sharemode */
352    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE, 0, &stg);
353    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
354    r = StgCreateDocfile( filename, STGM_SHARE_DENY_WRITE, 0, &stg);
355    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
356    r = StgCreateDocfile( filename, STGM_SHARE_DENY_NONE, 0, &stg);
357    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile failed\n");
358    r = StgCreateDocfile( filename, STGM_SHARE_DENY_NONE | STGM_TRANSACTED, 0, &stg);
359    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile failed\n");
360    r = StgCreateDocfile( filename, STGM_SHARE_DENY_NONE | STGM_READWRITE, 0, &stg);
361    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile failed\n");
362    r = StgCreateDocfile( filename, STGM_SHARE_DENY_NONE | STGM_WRITE, 0, &stg);
363    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile failed\n");
364    r = StgCreateDocfile( filename, STGM_SHARE_DENY_WRITE | STGM_WRITE, 0, &stg);
365    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile failed\n");
366    r = StgCreateDocfile( filename, STGM_SHARE_DENY_WRITE | STGM_READ, 0, &stg);
367    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile wrong error\n");
368    r = StgCreateDocfile( filename, STGM_TRANSACTED | STGM_SHARE_DENY_WRITE | STGM_READ, 0, &stg);
369    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile wrong error\n");
370    ok(DeleteFileA(filenameA), "failed to delete file\n");
371 
372    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
373    ok(r==S_OK, "StgCreateDocfile failed\n");
374    r = IStorage_Release(stg);
375    ok(r == 0, "storage not released\n");
376    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED |STGM_FAILIFTHERE, 0, &stg);
377    ok(r==STG_E_FILEALREADYEXISTS, "StgCreateDocfile wrong error\n");
378    r = StgCreateDocfile( filename, STGM_SHARE_EXCLUSIVE | STGM_WRITE, 0, &stg);
379    ok(r==STG_E_FILEALREADYEXISTS, "StgCreateDocfile wrong error\n");
380 
381    r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_DENY_WRITE | STGM_READWRITE, 0, &stg);
382    ok(r==STG_E_INVALIDFLAG, "StgCreateDocfile succeeded\n");
383    r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
384    ok(r==S_OK, "StgCreateDocfile failed\n");
385    r = IStorage_Release(stg);
386    ok(r == 0, "storage not released\n");
387    ok(DeleteFileA(filenameA), "failed to delete file\n");
388 
389    r = StgCreateDocfile( filename, STGM_CREATE | STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
390    ok(r==S_OK, "StgCreateDocfile failed\n");
391    r = IStorage_Release(stg);
392    ok(r == 0, "storage not released\n");
393    ok(DeleteFileA(filenameA), "failed to delete file\n");
394 
395    /* test the way excel uses StgCreateDocFile */
396    r = StgCreateDocfile( filename, STGM_TRANSACTED|STGM_CREATE|STGM_SHARE_DENY_WRITE|STGM_READWRITE, 0, &stg);
397    ok(r==S_OK, "StgCreateDocfile the excel way failed\n");
398    if(r == S_OK)
399    {
400       r = IStorage_Release(stg);
401       ok(r == 0, "storage not released\n");
402       ok(DeleteFileA(filenameA), "failed to delete file\n");
403    }
404 
405    /* and the way windows media uses it ... */
406    r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_DENY_NONE | STGM_READWRITE | STGM_TRANSACTED, 0, &stg);
407    ok(r==S_OK, "StgCreateDocfile the windows media way failed\n");
408    if (r == S_OK)
409    {
410       r = IStorage_Release(stg);
411       ok(r == 0, "storage not released\n");
412       ok(DeleteFileA(filenameA), "failed to delete file\n");
413    }
414 
415    /* looks like we need STGM_TRANSACTED or STGM_CREATE */
416    r = StgCreateDocfile( filename, STGM_TRANSACTED|STGM_SHARE_DENY_WRITE|STGM_READWRITE, 0, &stg);
417    ok(r==S_OK, "StgCreateDocfile the excel way failed\n");
418    if(r == S_OK)
419    {
420       r = IStorage_Release(stg);
421       ok(r == 0, "storage not released\n");
422       ok(DeleteFileA(filenameA), "failed to delete file\n");
423    }
424 
425    r = StgCreateDocfile( filename, STGM_TRANSACTED|STGM_CREATE|STGM_SHARE_DENY_WRITE|STGM_WRITE, 0, &stg);
426    ok(r==S_OK, "StgCreateDocfile the excel way failed\n");
427    if(r == S_OK)
428    {
429       r = IStorage_Release(stg);
430       ok(r == 0, "storage not released\n");
431       ok(DeleteFileA(filenameA), "failed to delete file\n");
432    }
433 
434    r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stg);
435    ok(r==S_OK, "StgCreateDocfile the powerpoint way failed\n");
436    if(r == S_OK)
437    {
438       r = IStorage_Release(stg);
439       ok(r == 0, "storage not released\n");
440       ok(DeleteFileA(filenameA), "failed to delete file\n");
441    }
442 
443    /* test the way msi uses StgCreateDocfile */
444    r = StgCreateDocfile( filename, STGM_DIRECT | STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &stg);
445    ok(r==S_OK, "StgCreateDocFile failed\n");
446    r = IStorage_Release(stg);
447    ok(r == 0, "storage not released\n");
448    ok(DeleteFileA(filenameA), "failed to delete file\n");
449 }
450 
test_stgcreatestorageex(void)451 static void test_stgcreatestorageex(void)
452 {
453    HRESULT (WINAPI *pStgCreateStorageEx)(const WCHAR* pwcsName, DWORD grfMode, DWORD stgfmt, DWORD grfAttrs, STGOPTIONS* pStgOptions, void* reserved, REFIID riid, void** ppObjectOpen);
454    HMODULE hOle32 = GetModuleHandleA("ole32");
455    IStorage *stg = NULL;
456    STGOPTIONS stgoptions = {1, 0, 4096};
457    HRESULT r;
458 
459    pStgCreateStorageEx = (void *) GetProcAddress(hOle32, "StgCreateStorageEx");
460    if (!pStgCreateStorageEx)
461    {
462       win_skip("skipping test on NT4\n");
463       return;
464    }
465 
466    DeleteFileA(filenameA);
467 
468    /* Verify that StgCreateStorageEx can accept an options param */
469    r = pStgCreateStorageEx( filename,
470                            STGM_SHARE_EXCLUSIVE | STGM_READWRITE,
471                            STGFMT_DOCFILE,
472                            0,
473                            &stgoptions,
474                            NULL,
475                            &IID_IStorage,
476                            (void **) &stg);
477    ok(r==S_OK || r==STG_E_UNIMPLEMENTEDFUNCTION, "StgCreateStorageEx with options failed\n");
478    if (r==STG_E_UNIMPLEMENTEDFUNCTION)
479    {
480       /* We're on win98 which means all bets are off.  Let's get out of here. */
481       win_skip("skipping test on win9x\n");
482       return;
483    }
484 
485    r = IStorage_Release(stg);
486    ok(r == 0, "storage not released\n");
487    ok(DeleteFileA(filenameA), "failed to delete file\n");
488 
489    /* Verify that StgCreateStorageEx can accept a NULL pStgOptions */
490    r = pStgCreateStorageEx( filename,
491                            STGM_SHARE_EXCLUSIVE | STGM_READWRITE,
492                            STGFMT_STORAGE,
493                            0,
494                            NULL,
495                            NULL,
496                            &IID_IStorage,
497                            (void **) &stg);
498    ok(r==S_OK, "StgCreateStorageEx with NULL options failed\n");
499    r = IStorage_Release(stg);
500    ok(r == 0, "storage not released\n");
501    ok(DeleteFileA(filenameA), "failed to delete file\n");
502 }
503 
test_storage_stream(void)504 static void test_storage_stream(void)
505 {
506     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
507     static const WCHAR longname[] = {
508         'a','a','a','a','a','a','a','a','a','a','a','a','a','a','a','a',
509         'a','a','a','a','a','a','a','a','a','a','a','a','a','a','a','a',0
510     };
511     IStorage *stg = NULL;
512     HRESULT r;
513     IStream *stm = NULL;
514     IStream *stm2 = NULL;
515     ULONG count = 0;
516     LARGE_INTEGER pos;
517     ULARGE_INTEGER p;
518     unsigned char buffer[0x100];
519     IUnknown *unk;
520     BOOL ret;
521 
522     DeleteFileA(filenameA);
523 
524     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
525     ok(r==S_OK, "StgCreateDocfile failed\n");
526 
527     /* try create some invalid streams */
528     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 1, 0, &stm );
529     ok(r==STG_E_INVALIDPARAMETER, "IStorage->CreateStream wrong error\n");
530     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 1, &stm );
531     ok(r==STG_E_INVALIDPARAMETER, "IStorage->CreateStream wrong error\n");
532     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, NULL );
533     ok(r==STG_E_INVALIDPOINTER, "IStorage->CreateStream wrong error\n");
534     r = IStorage_CreateStream(stg, NULL, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
535     ok(r==STG_E_INVALIDNAME, "IStorage->CreateStream wrong error\n");
536     r = IStorage_CreateStream(stg, longname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
537     ok(r==STG_E_INVALIDNAME || broken(r==S_OK) /* nt4 */,
538        "IStorage->CreateStream wrong error, got %d GetLastError()=%d\n", r, GetLastError());
539     r = IStorage_CreateStream(stg, stmname, STGM_READWRITE, 0, 0, &stm );
540     ok(r==STG_E_INVALIDFLAG, "IStorage->CreateStream wrong error\n");
541     r = IStorage_CreateStream(stg, stmname, STGM_READ, 0, 0, &stm );
542     ok(r==STG_E_INVALIDFLAG, "IStorage->CreateStream wrong error\n");
543     r = IStorage_CreateStream(stg, stmname, STGM_WRITE, 0, 0, &stm );
544     ok(r==STG_E_INVALIDFLAG, "IStorage->CreateStream wrong error\n");
545     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_DENY_NONE | STGM_READWRITE, 0, 0, &stm );
546     ok(r==STG_E_INVALIDFLAG, "IStorage->CreateStream wrong error\n");
547     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_DENY_NONE | STGM_READ, 0, 0, &stm );
548     ok(r==STG_E_INVALIDFLAG, "IStorage->CreateStream wrong error\n");
549 
550     /* now really create a stream and delete it */
551     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
552     ok(r==S_OK, "IStorage->CreateStream failed\n");
553 
554     /* test for support interfaces */
555     r = IStream_QueryInterface(stm, &IID_IPersist, (void**)&unk);
556     ok(r==E_NOINTERFACE, "got 0x%08x\n", r);
557     r = IStream_QueryInterface(stm, &IID_IPersistStream, (void**)&unk);
558     ok(r==E_NOINTERFACE, "got 0x%08x\n", r);
559 
560     r = IStream_Release(stm);
561     ok(r == 0, "wrong ref count\n");
562     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
563     ok(r==STG_E_FILEALREADYEXISTS, "IStorage->CreateStream failed\n");
564     r = IStorage_DestroyElement(stg,stmname);
565     ok(r==S_OK, "IStorage->DestroyElement failed\n");
566 
567     /* create a stream and write to it */
568     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
569     ok(r==S_OK, "IStorage->CreateStream failed\n");
570 
571     r = IStream_Clone(stm, &stm2);
572     ok(r==S_OK, "failed to clone stream\n");
573 
574     r = IStream_Write(stm, NULL, 0, NULL );
575     ok(r==STG_E_INVALIDPOINTER, "IStream->Write wrong error\n");
576     r = IStream_Write(stm, "Hello\n", 0, NULL );
577     ok(r==S_OK, "failed to write stream\n");
578     r = IStream_Write(stm, "Hello\n", 0, &count );
579     ok(r==S_OK, "failed to write stream\n");
580     r = IStream_Write(stm, "Hello\n", 6, &count );
581     ok(r==S_OK, "failed to write stream\n");
582     r = IStream_Commit(stm, STGC_DEFAULT );
583     ok(r==S_OK, "failed to commit stream\n");
584     r = IStream_Commit(stm, STGC_DEFAULT );
585     ok(r==S_OK, "failed to commit stream\n");
586 
587     /* Read past the end of the stream. */
588     pos.QuadPart = 3;
589     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p );
590     ok(r==S_OK, "failed to seek stream\n");
591     ok(p.QuadPart == 3, "at wrong place\n");
592     r = IStream_Read(stm, buffer, sizeof buffer, &count );
593     ok(r==S_OK, "failed to read\n");
594     ok(count == 3, "read bytes past end of stream\n");
595     pos.QuadPart = 10;
596     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p );
597     ok(r==S_OK, "failed to seek stream\n");
598     ok(p.QuadPart == 10, "at wrong place\n");
599     r = IStream_Read(stm, buffer, sizeof buffer, &count );
600     ok(r==S_OK, "failed to read\n");
601     ok(count == 0, "read bytes past end of stream\n");
602     pos.QuadPart = 10000;
603     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p );
604     ok(r==S_OK, "failed to seek stream\n");
605     ok(p.QuadPart == 10000, "at wrong place\n");
606     r = IStream_Read(stm, buffer, sizeof buffer, &count );
607     ok(r==S_OK, "failed to read\n");
608     ok(count == 0, "read bytes past end of stream\n");
609 
610     /* Convert to a big block stream, and read past the end. */
611     p.QuadPart = 5000;
612     r = IStream_SetSize(stm,p);
613     ok(r==S_OK, "failed to set pos\n");
614     pos.QuadPart = 4997;
615     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p );
616     ok(r==S_OK, "failed to seek stream\n");
617     ok(p.QuadPart == 4997, "at wrong place\n");
618     r = IStream_Read(stm, buffer, sizeof buffer, &count );
619     ok(r==S_OK, "failed to read\n");
620     ok(count == 3, "read bytes past end of stream\n");
621     pos.QuadPart = 5001;
622     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p );
623     ok(r==S_OK, "failed to seek stream\n");
624     ok(p.QuadPart == 5001, "at wrong place\n");
625     r = IStream_Read(stm, buffer, sizeof buffer, &count );
626     ok(r==S_OK, "failed to read\n");
627     ok(count == 0, "read bytes past end of stream\n");
628     pos.QuadPart = 10000;
629     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p );
630     ok(r==S_OK, "failed to seek stream\n");
631     ok(p.QuadPart == 10000, "at wrong place\n");
632     r = IStream_Read(stm, buffer, sizeof buffer, &count );
633     ok(r==S_OK, "failed to read\n");
634     ok(count == 0, "read bytes past end of stream\n");
635 
636     /* seek round a bit, reset the stream size */
637     pos.QuadPart = 0;
638     r = IStream_Seek(stm, pos, 3, &p );
639     ok(r==STG_E_INVALIDFUNCTION, "IStream->Seek returned wrong error\n");
640     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, NULL);
641     ok(r==S_OK, "failed to seek stream\n");
642     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p );
643     ok(r==S_OK, "failed to seek stream\n");
644     r = IStream_SetSize(stm,p);
645     ok(r==S_OK, "failed to set pos\n");
646     pos.QuadPart = 10;
647     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p );
648     ok(r==S_OK, "failed to seek stream\n");
649     ok(p.QuadPart == 10, "at wrong place\n");
650     r = IStream_Read(stm, buffer, sizeof buffer, &count );
651     ok(r==S_OK, "failed to set pos\n");
652     ok(count == 0, "read bytes from empty stream\n");
653     pos.QuadPart = 10000;
654     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p );
655     ok(r==S_OK, "failed to seek stream\n");
656     ok(p.QuadPart == 10000, "at wrong place\n");
657     r = IStream_Read(stm, buffer, sizeof buffer, &count );
658     ok(r==S_OK, "failed to set pos\n");
659     ok(count == 0, "read bytes from empty stream\n");
660     pos.QuadPart = 0;
661     r = IStream_Seek(stm, pos, STREAM_SEEK_END, &p );
662     ok(r==S_OK, "failed to seek stream\n");
663     ok(p.QuadPart == 0, "at wrong place\n");
664     r = IStream_Read(stm, buffer, sizeof buffer, &count );
665     ok(r==S_OK, "failed to set pos\n");
666     ok(count == 0, "read bytes from empty stream\n");
667 
668     /* wrap up */
669     r = IStream_Release(stm2);
670     ok(r == 0, "wrong ref count\n");
671 
672     /* create a stream and write to it */
673     r = IStorage_CreateStream(stg, stmname, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm2 );
674     ok(r==S_OK, "IStorage->CreateStream failed\n");
675 
676     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &p);
677     ok(r==STG_E_REVERTED, "overwritten stream should return STG_E_REVERTED instead of 0x%08x\n", r);
678 
679     r = IStream_Release(stm2);
680     ok(r == 0, "wrong ref count\n");
681     r = IStream_Release(stm);
682     ok(r == 0, "wrong ref count\n");
683 
684     r = IStorage_Release(stg);
685     ok(r == 0, "wrong ref count\n");
686 
687     /* try create some invalid streams */
688     stg = NULL;
689     stm = NULL;
690     r = StgOpenStorage(filename, NULL, STGM_READ | STGM_SHARE_DENY_WRITE, NULL, 0, &stg);
691     ok(r == S_OK, "should succeed\n");
692     if (stg)
693     {
694         r = IStorage_OpenStream(stg, stmname, NULL, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stm);
695         ok(r == STG_E_INVALIDFLAG, "IStorage->OpenStream should return STG_E_INVALIDFLAG instead of 0x%08x\n", r);
696         IStorage_Release(stg);
697     }
698 
699     ret = DeleteFileA(filenameA);
700     ok(ret, "file should exist\n");
701 }
702 
touch_file(LPCSTR filename)703 static BOOL touch_file(LPCSTR filename)
704 {
705     HANDLE file;
706 
707     file = CreateFileA(filename, GENERIC_READ|GENERIC_WRITE, 0, NULL,
708                 CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
709     if (file==INVALID_HANDLE_VALUE)
710         return FALSE;
711     CloseHandle(file);
712     return TRUE;
713 }
714 
is_zero_length(LPCSTR filename)715 static BOOL is_zero_length(LPCSTR filename)
716 {
717     HANDLE file;
718     DWORD len;
719 
720     file = CreateFileA(filename, GENERIC_READ, 0, NULL,
721                 OPEN_EXISTING, 0, NULL);
722     if (file==INVALID_HANDLE_VALUE)
723         return FALSE;
724     len = GetFileSize(file, NULL);
725     CloseHandle(file);
726     return len == 0;
727 }
728 
is_existing_file(LPCSTR filename)729 static BOOL is_existing_file(LPCSTR filename)
730 {
731     HANDLE file;
732 
733     file = CreateFileA(filename, GENERIC_READ, 0, NULL,
734                        OPEN_EXISTING, 0, NULL);
735     if (file==INVALID_HANDLE_VALUE)
736         return FALSE;
737     CloseHandle(file);
738     return TRUE;
739 }
740 
test_open_storage(void)741 static void test_open_storage(void)
742 {
743     static const WCHAR szNonExist[] = { 'n','o','n','e','x','i','s','t',0 };
744     IStorage *stg = NULL, *stg2 = NULL;
745     HRESULT r;
746     DWORD stgm;
747     BOOL ret;
748 
749     /* try opening a zero length file - it should stay zero length */
750     DeleteFileA(filenameA);
751     touch_file(filenameA);
752     stgm = STGM_NOSCRATCH | STGM_TRANSACTED | STGM_SHARE_DENY_WRITE | STGM_READWRITE;
753     r = StgOpenStorage( filename, NULL, stgm, NULL, 0, &stg);
754     ok(r==STG_E_FILEALREADYEXISTS, "StgOpenStorage didn't fail\n");
755 
756     stgm = STGM_SHARE_EXCLUSIVE | STGM_READWRITE;
757     r = StgOpenStorage( filename, NULL, stgm, NULL, 0, &stg);
758     ok(r==STG_E_FILEALREADYEXISTS, "StgOpenStorage didn't fail\n");
759     ok(is_zero_length(filenameA), "file length changed\n");
760 
761     DeleteFileA(filenameA);
762 
763     /* try opening a nonexistent file - it should not create it */
764     stgm = STGM_DIRECT | STGM_SHARE_EXCLUSIVE | STGM_READWRITE;
765     r = StgOpenStorage( filename, NULL, stgm, NULL, 0, &stg);
766     ok(r!=S_OK, "StgOpenStorage failed: 0x%08x\n", r);
767     if (r==S_OK) IStorage_Release(stg);
768     ok(!is_existing_file(filenameA), "StgOpenStorage should not create a file\n");
769     DeleteFileA(filenameA);
770 
771     /* create the file */
772     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
773     ok(r==S_OK, "StgCreateDocfile failed\n");
774     IStorage_Release(stg);
775 
776     r = StgOpenStorage( filename, NULL, 0, NULL, 0, &stg);
777     ok(r==STG_E_INVALIDFLAG, "StgOpenStorage wrong error\n");
778     r = StgOpenStorage( NULL, NULL, STGM_SHARE_EXCLUSIVE, NULL, 0, &stg);
779     ok(r==STG_E_INVALIDNAME, "StgOpenStorage wrong error\n");
780     r = StgOpenStorage( filename, NULL, STGM_SHARE_EXCLUSIVE | STGM_READ, NULL, 0, NULL);
781     ok(r==STG_E_INVALIDPOINTER, "StgOpenStorage wrong error\n");
782     r = StgOpenStorage( filename, NULL, STGM_SHARE_EXCLUSIVE | STGM_READ, NULL, 1, &stg);
783     ok(r==STG_E_INVALIDPARAMETER, "StgOpenStorage wrong error\n");
784     r = StgOpenStorage( szNonExist, NULL, STGM_SHARE_EXCLUSIVE | STGM_READ, NULL, 0, &stg);
785     ok(r==STG_E_FILENOTFOUND, "StgOpenStorage failed\n");
786     r = StgOpenStorage( filename, NULL, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READ, NULL, 0, &stg);
787     ok(r==STG_E_INVALIDFLAG, "StgOpenStorage failed\n");
788     r = StgOpenStorage( filename, NULL, STGM_SHARE_DENY_NONE | STGM_READ, NULL, 0, &stg);
789     ok(r==STG_E_INVALIDFLAG, "StgOpenStorage failed\n");
790     r = StgOpenStorage( filename, NULL, STGM_SHARE_DENY_READ | STGM_READ, NULL, 0, &stg);
791     ok(r==STG_E_INVALIDFLAG, "StgOpenStorage failed\n");
792     r = StgOpenStorage( filename, NULL, STGM_SHARE_DENY_WRITE | STGM_READWRITE, NULL, 0, &stg);
793     ok(r==STG_E_INVALIDFLAG, "StgOpenStorage failed\n");
794 
795     /* open it for real */
796     r = StgOpenStorage( filename, NULL, STGM_SHARE_DENY_NONE | STGM_READ | STGM_TRANSACTED, NULL, 0, &stg); /* XLViewer 97/2000 */
797     ok(r==S_OK, "StgOpenStorage failed\n");
798     if(stg)
799     {
800         r = IStorage_Release(stg);
801         ok(r == 0, "wrong ref count\n");
802     }
803 
804     r = StgOpenStorage( filename, NULL, STGM_SHARE_DENY_WRITE | STGM_READ, NULL, 0, &stg);
805     ok(r==S_OK, "StgOpenStorage failed\n");
806     if(stg)
807     {
808         r = IStorage_Release(stg);
809         ok(r == 0, "wrong ref count\n");
810     }
811 
812     /* test the way word opens its custom dictionary */
813     r = StgOpenStorage( filename, NULL, STGM_NOSCRATCH | STGM_TRANSACTED |
814                         STGM_SHARE_DENY_WRITE | STGM_READWRITE, NULL, 0, &stg);
815     ok(r==S_OK, "StgOpenStorage failed\n");
816     if(stg)
817     {
818         r = IStorage_Release(stg);
819         ok(r == 0, "wrong ref count\n");
820     }
821 
822     r = StgOpenStorage( filename, NULL, STGM_SHARE_EXCLUSIVE | STGM_READ, NULL, 0, &stg);
823     ok(r==S_OK, "StgOpenStorage failed\n");
824     r = StgOpenStorage( filename, NULL, STGM_SHARE_EXCLUSIVE | STGM_READ, NULL, 0, &stg2);
825     ok(r==STG_E_SHAREVIOLATION, "StgOpenStorage failed\n");
826     if(stg)
827     {
828         r = IStorage_Release(stg);
829         ok(r == 0, "wrong ref count\n");
830     }
831 
832     /* now try write to a storage file we opened read-only */
833     r = StgOpenStorage( filename, NULL, STGM_SHARE_EXCLUSIVE | STGM_READ, NULL, 0, &stg);
834     ok(r==S_OK, "StgOpenStorage failed\n");
835     if(stg)
836     {
837         static const WCHAR stmname[] =  { 'w','i','n','e','t','e','s','t',0};
838         IStream *stm = NULL;
839         IStorage *stg2 = NULL;
840 
841         r = IStorage_CreateStream( stg, stmname, STGM_WRITE | STGM_SHARE_EXCLUSIVE,
842                                    0, 0, &stm );
843         ok(r == STG_E_ACCESSDENIED, "CreateStream should fail\n");
844         r = IStorage_CreateStorage( stg, stmname, STGM_WRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
845         ok(r == STG_E_ACCESSDENIED, "CreateStream should fail\n");
846 
847         r = IStorage_Release(stg);
848         ok(r == 0, "wrong ref count\n");
849     }
850 
851     /* open like visio 2003 */
852     stg = NULL;
853     r = StgOpenStorage( filename, NULL, STGM_PRIORITY | STGM_SHARE_DENY_NONE, NULL, 0, &stg);
854     ok(r == S_OK, "should succeed\n");
855     if (stg)
856         IStorage_Release(stg);
857 
858     /* test other sharing modes with STGM_PRIORITY */
859     stg = NULL;
860     r = StgOpenStorage( filename, NULL, STGM_PRIORITY | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg);
861     ok(r == S_OK, "should succeed\n");
862     if (stg)
863         IStorage_Release(stg);
864 
865     stg = NULL;
866     r = StgOpenStorage( filename, NULL, STGM_PRIORITY | STGM_SHARE_DENY_WRITE, NULL, 0, &stg);
867     ok(r == S_OK, "should succeed\n");
868     if (stg)
869         IStorage_Release(stg);
870 
871     stg = NULL;
872     r = StgOpenStorage( filename, NULL, STGM_PRIORITY | STGM_SHARE_DENY_READ, NULL, 0, &stg);
873     ok(r == S_OK, "should succeed\n");
874     if (stg)
875         IStorage_Release(stg);
876 
877     /* open like Project 2003 */
878     stg = NULL;
879     r = StgOpenStorage( filename, NULL, STGM_PRIORITY, NULL, 0, &stg);
880     ok(r == S_OK, "should succeed\n");
881     r = StgOpenStorage( filename, NULL, STGM_PRIORITY, NULL, 0, &stg2);
882     ok(r == S_OK, "should succeed\n");
883     if (stg2)
884         IStorage_Release(stg2);
885     if (stg)
886         IStorage_Release(stg);
887 
888     stg = NULL;
889     r = StgOpenStorage( filename, NULL, STGM_PRIORITY | STGM_READWRITE, NULL, 0, &stg);
890     ok(r == STG_E_INVALIDFLAG, "should fail\n");
891 
892     r = StgOpenStorage( filename, NULL, STGM_TRANSACTED | STGM_PRIORITY, NULL, 0, &stg);
893     ok(r == STG_E_INVALIDFLAG, "should fail\n");
894 
895     r = StgOpenStorage( filename, NULL, STGM_SIMPLE | STGM_PRIORITY, NULL, 0, &stg);
896     ok(r == STG_E_INVALIDFLAG, "should fail\n");
897 
898     r = StgOpenStorage( filename, NULL, STGM_DELETEONRELEASE | STGM_PRIORITY, NULL, 0, &stg);
899     ok(r == STG_E_INVALIDFUNCTION, "should fail\n");
900 
901     r = StgOpenStorage( filename, NULL, STGM_NOSCRATCH | STGM_PRIORITY, NULL, 0, &stg);
902     ok(r == STG_E_INVALIDFLAG, "should fail\n");
903 
904     r = StgOpenStorage( filename, NULL, STGM_NOSNAPSHOT | STGM_PRIORITY, NULL, 0, &stg);
905     ok(r == STG_E_INVALIDFLAG, "should fail\n");
906 
907     ret = DeleteFileA(filenameA);
908     ok(ret, "file didn't exist\n");
909 }
910 
test_storage_suminfo(void)911 static void test_storage_suminfo(void)
912 {
913     IStorage *stg = NULL;
914     IPropertySetStorage *propset = NULL;
915     IPropertyStorage *ps = NULL;
916     HRESULT r;
917 
918     DeleteFileA(filenameA);
919 
920     /* create the file */
921     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
922                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
923     ok(r==S_OK, "StgCreateDocfile failed\n");
924 
925     r = IStorage_QueryInterface( stg, &IID_IPropertySetStorage, (LPVOID) &propset );
926     ok(r == S_OK, "query interface failed\n");
927 
928     /* delete it */
929     r = IPropertySetStorage_Delete( propset, &FMTID_SummaryInformation );
930     ok(r == STG_E_FILENOTFOUND, "deleted property set storage\n");
931 
932     r = IPropertySetStorage_Open( propset, &FMTID_SummaryInformation,
933                                 STGM_READ | STGM_SHARE_EXCLUSIVE, &ps );
934     ok(r == STG_E_FILENOTFOUND, "opened property set storage\n");
935 
936     r = IPropertySetStorage_Create( propset, &FMTID_SummaryInformation, NULL, 0,
937                                 STGM_READ | STGM_SHARE_EXCLUSIVE, &ps );
938     ok(r == STG_E_INVALIDFLAG, "created property set storage\n");
939 
940     r = IPropertySetStorage_Create( propset, &FMTID_SummaryInformation, NULL, 0,
941                                 STGM_READ, &ps );
942     ok(r == STG_E_INVALIDFLAG, "created property set storage\n");
943 
944     r = IPropertySetStorage_Create( propset, &FMTID_SummaryInformation, NULL, 0, 0, &ps );
945     ok(r == STG_E_INVALIDFLAG, "created property set storage\n");
946 
947     r = IPropertySetStorage_Create( propset, &FMTID_SummaryInformation, NULL, 0,
948                                 STGM_WRITE|STGM_SHARE_EXCLUSIVE, &ps );
949     ok(r == STG_E_INVALIDFLAG, "created property set storage\n");
950 
951     r = IPropertySetStorage_Create( propset, &FMTID_SummaryInformation, NULL, 0,
952                                 STGM_CREATE|STGM_WRITE|STGM_SHARE_EXCLUSIVE, &ps );
953     ok(r == STG_E_INVALIDFLAG, "created property set storage\n");
954 
955     /* now try really creating a property set */
956     r = IPropertySetStorage_Create( propset, &FMTID_SummaryInformation, NULL, 0,
957                                 STGM_CREATE|STGM_READWRITE|STGM_SHARE_EXCLUSIVE, &ps );
958     ok(r == S_OK, "failed to create property set storage\n");
959 
960     if( ps )
961         IPropertyStorage_Release(ps);
962 
963     /* now try creating the same thing again */
964     r = IPropertySetStorage_Create( propset, &FMTID_SummaryInformation, NULL, 0,
965                                 STGM_CREATE|STGM_READWRITE|STGM_SHARE_EXCLUSIVE, &ps );
966     ok(r == S_OK, "failed to create property set storage\n");
967     if( ps )
968         IPropertyStorage_Release(ps);
969 
970     /* should be able to open it */
971     r = IPropertySetStorage_Open( propset, &FMTID_SummaryInformation,
972             STGM_READWRITE|STGM_SHARE_EXCLUSIVE, &ps);
973     ok(r == S_OK, "open failed\n");
974     if(r == S_OK)
975         IPropertyStorage_Release(ps);
976 
977     /* delete it */
978     r = IPropertySetStorage_Delete( propset, &FMTID_SummaryInformation );
979     ok(r == S_OK, "failed to delete property set storage\n");
980 
981     /* try opening with an invalid FMTID */
982     r = IPropertySetStorage_Open( propset, NULL,
983             STGM_READWRITE|STGM_SHARE_EXCLUSIVE, &ps);
984     ok(r == E_INVALIDARG, "open succeeded\n");
985     if(r == S_OK)
986         IPropertyStorage_Release(ps);
987 
988     /* try a bad guid */
989     r = IPropertySetStorage_Open( propset, &IID_IStorage,
990             STGM_READWRITE|STGM_SHARE_EXCLUSIVE, &ps);
991     ok(r == STG_E_FILENOTFOUND, "open succeeded\n");
992     if(r == S_OK)
993         IPropertyStorage_Release(ps);
994 
995 
996     /* try some invalid flags */
997     r = IPropertySetStorage_Open( propset, &FMTID_SummaryInformation,
998             STGM_CREATE | STGM_READWRITE|STGM_SHARE_EXCLUSIVE, &ps);
999     ok(r == STG_E_INVALIDFLAG, "open succeeded\n");
1000     if(r == S_OK)
1001         IPropertyStorage_Release(ps);
1002 
1003     /* after deleting it, it should be gone */
1004     r = IPropertySetStorage_Open( propset, &FMTID_SummaryInformation,
1005             STGM_READWRITE|STGM_SHARE_EXCLUSIVE, &ps);
1006     ok(r == STG_E_FILENOTFOUND, "open failed\n");
1007     if(r == S_OK)
1008         IPropertyStorage_Release(ps);
1009 
1010     r = IPropertySetStorage_Release( propset );
1011     ok(r == 1, "ref count wrong\n");
1012 
1013     r = IStorage_Release(stg);
1014     ok(r == 0, "ref count wrong\n");
1015 
1016     DeleteFileA(filenameA);
1017 }
1018 
test_storage_refcount(void)1019 static void test_storage_refcount(void)
1020 {
1021     IStorage *stg = NULL;
1022     IStorage *stgprio = NULL;
1023     HRESULT r;
1024     IStream *stm = NULL;
1025     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
1026     LARGE_INTEGER pos;
1027     ULARGE_INTEGER upos;
1028     STATSTG stat;
1029     char buffer[10];
1030 
1031     DeleteFileA(filenameA);
1032 
1033     /* create the file */
1034     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1035                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
1036     ok(r==S_OK, "StgCreateDocfile failed\n");
1037 
1038     r = WriteClassStg( stg, &test_stg_cls );
1039     ok( r == S_OK, "WriteClassStg failed\n");
1040 
1041     r = IStorage_Commit( stg, STGC_DEFAULT );
1042     ok( r == S_OK, "IStorage_Commit failed\n");
1043 
1044     /* now create a stream */
1045     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1046     ok(r==S_OK, "IStorage->CreateStream failed\n");
1047 
1048     r = IStorage_Release( stg );
1049     ok (r == 0, "storage not released\n");
1050 
1051     pos.QuadPart = 0;
1052     r = IStream_Seek( stm, pos, 0, &upos );
1053     ok (r == STG_E_REVERTED, "seek should fail\n");
1054 
1055     r = IStream_Stat( stm, &stat, STATFLAG_DEFAULT );
1056     ok (r == STG_E_REVERTED, "stat should fail\n");
1057 
1058     r = IStream_Write( stm, "Test string", strlen("Test string"), NULL);
1059     ok (r == STG_E_REVERTED, "IStream_Write should return STG_E_REVERTED instead of 0x%08x\n", r);
1060 
1061     r = IStream_Read( stm, buffer, sizeof(buffer), NULL);
1062     ok (r == STG_E_REVERTED, "IStream_Read should return STG_E_REVERTED instead of 0x%08x\n", r);
1063 
1064     r = IStream_Release(stm);
1065     ok (r == 0, "stream not released\n");
1066 
1067     /* tests that STGM_PRIORITY doesn't prevent readwrite access from other
1068      * StgOpenStorage calls in transacted mode */
1069     r = StgOpenStorage( filename, NULL, STGM_PRIORITY, NULL, 0, &stgprio);
1070     ok(r==S_OK, "StgOpenStorage failed with error 0x%08x\n", r);
1071 
1072     /* non-transacted mode read/write fails */
1073     r = StgOpenStorage( filename, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg);
1074     ok(r==STG_E_LOCKVIOLATION, "StgOpenStorage should return STG_E_LOCKVIOLATION instead of 0x%08x\n", r);
1075 
1076     /* non-transacted mode read-only succeeds */
1077     r = StgOpenStorage( filename, NULL, STGM_SHARE_DENY_WRITE|STGM_READ, NULL, 0, &stg);
1078     ok(r==S_OK, "StgOpenStorage failed with error 0x%08x\n", r);
1079     IStorage_Release(stg);
1080 
1081     r = StgOpenStorage( filename, NULL, STGM_TRANSACTED|STGM_SHARE_DENY_WRITE|STGM_READWRITE, NULL, 0, &stg);
1082     ok(r==S_OK, "StgOpenStorage failed with error 0x%08x\n", r);
1083     if(stg)
1084     {
1085         static const WCHAR stgname[] = { ' ',' ',' ','2','9',0 };
1086         static const WCHAR stgname2[] = { 'C','V','_','i','e','w',0 };
1087         static const WCHAR stmname2[] = { 'V','a','r','2','D','a','t','a',0 };
1088         IStorage *stg2;
1089         IStorage *stg3;
1090         STATSTG statstg;
1091 
1092         r = IStorage_Stat( stg, &statstg, STATFLAG_NONAME );
1093         ok(r == S_OK, "Stat should have succeeded instead of returning 0x%08x\n", r);
1094         ok(statstg.type == STGTY_STORAGE, "Statstg type should have been STGTY_STORAGE instead of %d\n", statstg.type);
1095         ok(U(statstg.cbSize).LowPart == 0, "Statstg cbSize.LowPart should have been 0 instead of %d\n", U(statstg.cbSize).LowPart);
1096         ok(U(statstg.cbSize).HighPart == 0, "Statstg cbSize.HighPart should have been 0 instead of %d\n", U(statstg.cbSize).HighPart);
1097         ok(statstg.grfMode == (STGM_TRANSACTED|STGM_SHARE_DENY_WRITE|STGM_READWRITE),
1098             "Statstg grfMode should have been 0x10022 instead of 0x%x\n", statstg.grfMode);
1099         ok(statstg.grfLocksSupported == 0, "Statstg grfLocksSupported should have been 0 instead of %d\n", statstg.grfLocksSupported);
1100         ok(IsEqualCLSID(&statstg.clsid, &test_stg_cls), "Statstg clsid is not test_stg_cls\n");
1101         ok(statstg.grfStateBits == 0, "Statstg grfStateBits should have been 0 instead of %d\n", statstg.grfStateBits);
1102         ok(statstg.reserved == 0, "Statstg reserved should have been 0 instead of %d\n", statstg.reserved);
1103 
1104         r = IStorage_CreateStorage( stg, stgname, STGM_SHARE_EXCLUSIVE, 0, 0, &stg2 );
1105         ok(r == S_OK, "CreateStorage should have succeeded instead of returning 0x%08x\n", r);
1106 
1107         r = IStorage_Stat( stg2, &statstg, STATFLAG_DEFAULT );
1108         ok(r == S_OK, "Stat should have succeeded instead of returning 0x%08x\n", r);
1109         ok(!memcmp(statstg.pwcsName, stgname, sizeof(stgname)),
1110             "Statstg pwcsName should have been the name the storage was created with\n");
1111         ok(statstg.type == STGTY_STORAGE, "Statstg type should have been STGTY_STORAGE instead of %d\n", statstg.type);
1112         ok(U(statstg.cbSize).LowPart == 0, "Statstg cbSize.LowPart should have been 0 instead of %d\n", U(statstg.cbSize).LowPart);
1113         ok(U(statstg.cbSize).HighPart == 0, "Statstg cbSize.HighPart should have been 0 instead of %d\n", U(statstg.cbSize).HighPart);
1114         ok(statstg.grfMode == STGM_SHARE_EXCLUSIVE,
1115             "Statstg grfMode should have been STGM_SHARE_EXCLUSIVE instead of 0x%x\n", statstg.grfMode);
1116         ok(statstg.grfLocksSupported == 0, "Statstg grfLocksSupported should have been 0 instead of %d\n", statstg.grfLocksSupported);
1117         ok(IsEqualCLSID(&statstg.clsid, &CLSID_NULL), "Statstg clsid is not CLSID_NULL\n");
1118         ok(statstg.grfStateBits == 0, "Statstg grfStateBits should have been 0 instead of %d\n", statstg.grfStateBits);
1119         ok(statstg.reserved == 0, "Statstg reserved should have been 0 instead of %d\n", statstg.reserved);
1120         CoTaskMemFree(statstg.pwcsName);
1121 
1122         r = IStorage_CreateStorage( stg2, stgname2, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, 0, &stg3 );
1123         ok(r == STG_E_ACCESSDENIED, "CreateStorage should have returned STG_E_ACCESSDENIED instead of 0x%08x\n", r);
1124 
1125         r = IStorage_CreateStream( stg2, stmname2, STGM_CREATE|STGM_SHARE_EXCLUSIVE, 0, 0, &stm );
1126         ok(r == STG_E_ACCESSDENIED, "CreateStream should have returned STG_E_ACCESSDENIED instead of 0x%08x\n", r);
1127 
1128         IStorage_Release(stg2);
1129 
1130         r = IStorage_Release(stg);
1131         ok(r == 0, "wrong ref count\n");
1132     }
1133 
1134     /* Multiple STGM_PRIORITY opens are possible. */
1135     r = StgOpenStorage( filename, NULL, STGM_PRIORITY, NULL, 0, &stg);
1136     ok(r==S_OK, "StgOpenStorage failed with error 0x%08x\n", r);
1137     if(stg)
1138     {
1139         r = IStorage_Release(stg);
1140         ok(r == 0, "wrong ref count\n");
1141     }
1142 
1143     r = StgOpenStorage( NULL, stgprio, STGM_TRANSACTED|STGM_SHARE_DENY_WRITE|STGM_READWRITE, NULL, 0, &stg);
1144     ok(r==S_OK, "StgOpenStorage failed with error 0x%08x\n", r);
1145     if(stg)
1146     {
1147         static const WCHAR stgname[] = { ' ',' ',' ','2','9',0 };
1148         IStorage *stg2;
1149         STATSTG statstg;
1150 
1151         r = IStorage_Stat( stg, &statstg, STATFLAG_NONAME );
1152         ok(r == S_OK, "Stat should have succeeded instead of returning 0x%08x\n", r);
1153         ok(statstg.type == STGTY_STORAGE, "Statstg type should have been STGTY_STORAGE instead of %d\n", statstg.type);
1154         ok(U(statstg.cbSize).LowPart == 0, "Statstg cbSize.LowPart should have been 0 instead of %d\n", U(statstg.cbSize).LowPart);
1155         ok(U(statstg.cbSize).HighPart == 0, "Statstg cbSize.HighPart should have been 0 instead of %d\n", U(statstg.cbSize).HighPart);
1156         ok(statstg.grfMode == (STGM_TRANSACTED|STGM_SHARE_DENY_WRITE|STGM_READWRITE),
1157             "Statstg grfMode should have been 0x10022 instead of 0x%x\n", statstg.grfMode);
1158         ok(statstg.grfLocksSupported == 0, "Statstg grfLocksSupported should have been 0 instead of %d\n", statstg.grfLocksSupported);
1159         ok(IsEqualCLSID(&statstg.clsid, &test_stg_cls), "Statstg clsid is not test_stg_cls\n");
1160         ok(statstg.grfStateBits == 0, "Statstg grfStateBits should have been 0 instead of %d\n", statstg.grfStateBits);
1161         ok(statstg.reserved == 0, "Statstg reserved should have been 0 instead of %d\n", statstg.reserved);
1162 
1163         r = IStorage_CreateStorage( stg, stgname, STGM_SHARE_EXCLUSIVE, 0, 0, &stg2 );
1164         ok(r == S_OK, "CreateStorage should have succeeded instead of returning 0x%08x\n", r);
1165 
1166         IStorage_Release(stg2);
1167 
1168         r = IStorage_Commit( stg, 0 );
1169         ok(r == S_OK, "Commit should have succeeded instead of returning 0x%08x\n", r);
1170 
1171         r = IStorage_Release(stg);
1172         ok(r == 0, "wrong ref count\n");
1173     }
1174     /* IStorage_Release(stgprio) not necessary because StgOpenStorage released it. */
1175 
1176     DeleteFileA(filenameA);
1177 }
1178 
test_writeclassstg(void)1179 static void test_writeclassstg(void)
1180 {
1181     IStorage *stg = NULL;
1182     HRESULT r;
1183     CLSID temp_cls, cls2;
1184 
1185     DeleteFileA(filenameA);
1186 
1187     /* create the file */
1188     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1189                             STGM_READWRITE, 0, &stg);
1190     ok(r==S_OK, "StgCreateDocfile failed\n");
1191 
1192     r = ReadClassStg( NULL, NULL );
1193     ok(r == E_INVALIDARG, "ReadClassStg should return E_INVALIDARG instead of 0x%08X\n", r);
1194 
1195     memset(&temp_cls, 0xcc, sizeof(temp_cls));
1196     memset(&cls2, 0xcc, sizeof(cls2));
1197     r = ReadClassStg( NULL, &temp_cls );
1198     ok(r == E_INVALIDARG, "got 0x%08x\n", r);
1199     ok(IsEqualCLSID(&temp_cls, &cls2), "got wrong clsid\n");
1200 
1201     r = ReadClassStg( stg, NULL );
1202     ok(r == E_INVALIDARG, "ReadClassStg should return E_INVALIDARG instead of 0x%08X\n", r);
1203 
1204     temp_cls.Data1 = 0xdeadbeef;
1205     r = ReadClassStg( stg, &temp_cls );
1206     ok(r == S_OK, "ReadClassStg failed with 0x%08X\n", r);
1207 
1208     ok(IsEqualCLSID(&temp_cls, &CLSID_NULL), "ReadClassStg returned wrong clsid\n");
1209 
1210     r = WriteClassStg( NULL, NULL );
1211     ok(r == E_INVALIDARG, "WriteClassStg should return E_INVALIDARG instead of 0x%08X\n", r);
1212 
1213     r = WriteClassStg( stg, NULL );
1214     ok(r == STG_E_INVALIDPOINTER, "WriteClassStg should return STG_E_INVALIDPOINTER instead of 0x%08X\n", r);
1215 
1216     r = WriteClassStg( stg, &test_stg_cls );
1217     ok( r == S_OK, "WriteClassStg failed with 0x%08X\n", r);
1218 
1219     r = ReadClassStg( stg, &temp_cls );
1220     ok( r == S_OK, "ReadClassStg failed with 0x%08X\n", r);
1221     ok(IsEqualCLSID(&temp_cls, &test_stg_cls), "ReadClassStg returned wrong clsid\n");
1222 
1223     r = IStorage_Release( stg );
1224     ok (r == 0, "storage not released\n");
1225 
1226     DeleteFileA(filenameA);
1227 }
1228 
test_streamenum(void)1229 static void test_streamenum(void)
1230 {
1231     IStorage *stg = NULL;
1232     HRESULT r;
1233     IStream *stm = NULL;
1234     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
1235     static const WCHAR stmname2[] = { 'A','B','C','D','E','F','G','H','I',0 };
1236     static const WCHAR stmname3[] = { 'A','B','C','D','E','F','G','H','I','J',0 };
1237     static const STATSTG stat_null;
1238     STATSTG stat;
1239     IEnumSTATSTG *ee = NULL;
1240     ULONG count;
1241 
1242     DeleteFileA(filenameA);
1243 
1244     /* create the file */
1245     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1246                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
1247     ok(r==S_OK, "StgCreateDocfile failed\n");
1248 
1249     r = WriteClassStg( stg, &test_stg_cls );
1250     ok( r == S_OK, "WriteClassStg failed\n");
1251 
1252     r = IStorage_Commit( stg, STGC_DEFAULT );
1253     ok( r == S_OK, "IStorage_Commit failed\n");
1254 
1255     /* now create a stream */
1256     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1257     ok(r==S_OK, "IStorage->CreateStream failed\n");
1258 
1259     IStream_Release(stm);
1260 
1261     /* first enum ... should be 1 stream */
1262     r = IStorage_EnumElements(stg, 0, NULL, 0, &ee);
1263     ok(r==S_OK, "IStorage->EnumElements failed\n");
1264 
1265     count = 0xf00;
1266     r = IEnumSTATSTG_Next(ee, 1, &stat, &count);
1267     ok(r==S_OK, "IEnumSTATSTG->Next failed\n");
1268     ok(count == 1, "count wrong\n");
1269 
1270     if (r == S_OK)
1271         CoTaskMemFree(stat.pwcsName);
1272 
1273     r = IEnumSTATSTG_Release(ee);
1274     ok(r==S_OK, "EnumSTATSTG_Release failed with error 0x%08x\n", r);
1275 
1276     /* second enum... destroy the stream before reading */
1277     r = IStorage_EnumElements(stg, 0, NULL, 0, &ee);
1278     ok(r==S_OK, "IStorage->EnumElements failed\n");
1279 
1280     r = IStorage_DestroyElement(stg, stmname);
1281     ok(r==S_OK, "IStorage->DestroyElement failed\n");
1282 
1283     memset(&stat, 0xad, sizeof(stat));
1284     count = 0xf00;
1285     r = IEnumSTATSTG_Next(ee, 1, &stat, &count);
1286     ok(r==S_FALSE, "IEnumSTATSTG->Next failed\n");
1287     ok(count == 0, "count wrong\n");
1288     ok(memcmp(&stat, &stat_null, sizeof(stat)) == 0, "stat is not zeroed\n");
1289 
1290     /* reset and try again */
1291     r = IEnumSTATSTG_Reset(ee);
1292     ok(r==S_OK, "IEnumSTATSTG->Reset failed\n");
1293 
1294     count = 0xf00;
1295     r = IEnumSTATSTG_Next(ee, 1, &stat, &count);
1296     ok(r==S_FALSE, "IEnumSTATSTG->Next failed\n");
1297     ok(count == 0, "count wrong\n");
1298 
1299     /* add a stream before reading */
1300     r = IEnumSTATSTG_Reset(ee);
1301     ok(r==S_OK, "IEnumSTATSTG->Reset failed\n");
1302 
1303     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1304     ok(r==S_OK, "IStorage->CreateStream failed\n");
1305 
1306     r = IStream_Release(stm);
1307     ok(r==S_OK, "Stream_Release failed with error 0x%08x\n", r);
1308 
1309     count = 0xf00;
1310     r = IEnumSTATSTG_Next(ee, 1, &stat, &count);
1311     ok(r==S_OK, "IEnumSTATSTG->Next failed\n");
1312     ok(count == 1, "count wrong\n");
1313 
1314     if (r == S_OK)
1315     {
1316         ok(lstrcmpiW(stat.pwcsName, stmname) == 0, "expected CONTENTS, got %s\n", wine_dbgstr_w(stat.pwcsName));
1317         CoTaskMemFree(stat.pwcsName);
1318     }
1319 
1320     r = IStorage_CreateStream(stg, stmname2, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1321     ok(r==S_OK, "IStorage->CreateStream failed\n");
1322 
1323     r = IStream_Release(stm);
1324     ok(r==S_OK, "Stream_Release failed with error 0x%08x\n", r);
1325 
1326     count = 0xf00;
1327     r = IEnumSTATSTG_Next(ee, 1, &stat, &count);
1328     ok(r==S_OK, "IEnumSTATSTG->Next failed\n");
1329     ok(count == 1, "count wrong\n");
1330 
1331     if (r == S_OK)
1332     {
1333         ok(lstrcmpiW(stat.pwcsName, stmname2) == 0, "expected ABCDEFGHI, got %s\n", wine_dbgstr_w(stat.pwcsName));
1334         CoTaskMemFree(stat.pwcsName);
1335     }
1336 
1337     /* delete previous and next stream after reading */
1338     r = IStorage_CreateStream(stg, stmname3, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1339     ok(r==S_OK, "IStorage->CreateStream failed\n");
1340 
1341     r = IStream_Release(stm);
1342     ok(r==S_OK, "Stream_Release failed with error 0x%08x\n", r);
1343 
1344     r = IEnumSTATSTG_Reset(ee);
1345     ok(r==S_OK, "IEnumSTATSTG->Reset failed\n");
1346 
1347     count = 0xf00;
1348     r = IEnumSTATSTG_Next(ee, 1, &stat, &count);
1349     ok(r==S_OK, "IEnumSTATSTG->Next failed\n");
1350     ok(count == 1, "count wrong\n");
1351 
1352     if (r == S_OK)
1353     {
1354         ok(lstrcmpiW(stat.pwcsName, stmname) == 0, "expected CONTENTS, got %s\n", wine_dbgstr_w(stat.pwcsName));
1355         CoTaskMemFree(stat.pwcsName);
1356     }
1357 
1358     r = IStorage_DestroyElement(stg, stmname);
1359     ok(r==S_OK, "IStorage->DestroyElement failed\n");
1360 
1361     r = IStorage_DestroyElement(stg, stmname2);
1362     ok(r==S_OK, "IStorage->DestroyElement failed\n");
1363 
1364     count = 0xf00;
1365     r = IEnumSTATSTG_Next(ee, 1, &stat, &count);
1366     ok(r==S_OK, "IEnumSTATSTG->Next failed\n");
1367     ok(count == 1, "count wrong\n");
1368 
1369     if (r == S_OK)
1370     {
1371         ok(lstrcmpiW(stat.pwcsName, stmname3) == 0, "expected ABCDEFGHIJ, got %s\n", wine_dbgstr_w(stat.pwcsName));
1372         CoTaskMemFree(stat.pwcsName);
1373     }
1374 
1375     r = IStorage_Release( stg );
1376     todo_wine ok (r == 0, "storage not released\n");
1377 
1378     /* enumerator is still valid and working after the storage is released */
1379     r = IEnumSTATSTG_Reset(ee);
1380     ok(r==S_OK, "IEnumSTATSTG->Reset failed\n");
1381 
1382     count = 0xf00;
1383     r = IEnumSTATSTG_Next(ee, 1, &stat, &count);
1384     ok(r==S_OK, "IEnumSTATSTG->Next failed\n");
1385     ok(count == 1, "count wrong\n");
1386 
1387     if (r == S_OK)
1388     {
1389         ok(lstrcmpiW(stat.pwcsName, stmname3) == 0, "expected ABCDEFGHIJ, got %s\n", wine_dbgstr_w(stat.pwcsName));
1390         CoTaskMemFree(stat.pwcsName);
1391     }
1392 
1393     /* the storage is left open until the enumerator is freed */
1394     r = StgOpenStorage( filename, NULL, STGM_SHARE_EXCLUSIVE |
1395                             STGM_READWRITE |STGM_TRANSACTED, NULL, 0, &stg);
1396     ok(r==STG_E_SHAREVIOLATION ||
1397        r==STG_E_LOCKVIOLATION, /* XP-SP2/W2K3-SP1 and below */
1398        "StgCreateDocfile failed, res=%x\n", r);
1399 
1400     r = IEnumSTATSTG_Release(ee);
1401     ok (r == 0, "enum not released\n");
1402 
1403     DeleteFileA(filenameA);
1404 }
1405 
test_transact(void)1406 static void test_transact(void)
1407 {
1408     IStorage *stg = NULL, *stg2 = NULL, *stg3 = NULL;
1409     HRESULT r;
1410     IStream *stm = NULL;
1411     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
1412     static const WCHAR stmname2[] = { 'F','O','O',0 };
1413     static const WCHAR stgname[] = { 'P','E','R','M','S','T','G',0 };
1414     static const WCHAR stgname2[] = { 'T','E','M','P','S','T','G',0 };
1415     BOOL ret;
1416 
1417     DeleteFileA(filenameA);
1418 
1419     /* create the file */
1420     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1421                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
1422     ok(r==S_OK, "StgCreateDocfile failed\n");
1423 
1424     /* commit a new stream and storage */
1425     r = IStorage_CreateStream(stg, stmname2, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1426     ok(r==S_OK, "IStorage->CreateStream failed\n");
1427 
1428     r = IStream_Write(stm, "this is stream 1\n", 16, NULL);
1429     ok(r==S_OK, "IStream->Write failed\n");
1430 
1431     IStream_Release(stm);
1432 
1433     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
1434     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1435 
1436     if (r == S_OK)
1437     {
1438         /* Create two substorages but only commit one */
1439         r = IStorage_CreateStorage(stg2, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg3);
1440         ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1441 
1442         if (r == S_OK)
1443             IStorage_Release(stg3);
1444 
1445         r = IStorage_Commit(stg, 0);
1446         ok(r==S_OK, "IStorage->Commit failed\n");
1447 
1448         r = IStorage_CreateStorage(stg2, stgname2, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg3);
1449         ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1450 
1451         if (r == S_OK)
1452             IStorage_Release(stg3);
1453 
1454         IStorage_Release(stg2);
1455     }
1456 
1457     /* now create a stream and storage, but don't commit them */
1458     stm = NULL;
1459     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1460     ok(r==S_OK, "IStorage->CreateStream failed\n");
1461 
1462     r = IStream_Write(stm, "this is stream 2\n", 16, NULL);
1463     ok(r==S_OK, "IStream->Write failed\n");
1464 
1465     /* IStream::Commit does nothing for OLE storage streams */
1466     r = IStream_Commit(stm, STGC_ONLYIFCURRENT | STGC_DANGEROUSLYCOMMITMERELYTODISKCACHE);
1467     ok(r==S_OK, "IStream->Commit failed\n");
1468 
1469     r = IStorage_CreateStorage(stg, stgname2, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
1470     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1471 
1472     if (r == S_OK)
1473         IStorage_Release(stg2);
1474 
1475     IStream_Release(stm);
1476 
1477     IStorage_Release(stg);
1478 
1479     stm = NULL;
1480     stg = NULL;
1481     r = StgOpenStorage( filename, NULL, STGM_SHARE_DENY_NONE | STGM_READ | STGM_TRANSACTED, NULL, 0, &stg);
1482     ok(r==S_OK, "StgOpenStorage failed\n");
1483 
1484     if (!stg)
1485         return;
1486 
1487     r = IStorage_OpenStream(stg, stmname, NULL, STGM_SHARE_DENY_NONE|STGM_READ, 0, &stm );
1488     ok(r==STG_E_INVALIDFLAG, "IStorage->OpenStream failed %08x\n", r);
1489 
1490     r = IStorage_OpenStream(stg, stmname, NULL, STGM_DELETEONRELEASE|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1491     ok(r==STG_E_INVALIDFUNCTION, "IStorage->OpenStream failed %08x\n", r);
1492 
1493     r = IStorage_OpenStream(stg, stmname, NULL, STGM_TRANSACTED|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1494     ok(r==STG_E_INVALIDFUNCTION, "IStorage->OpenStream failed %08x\n", r);
1495 
1496     r = IStorage_OpenStorage(stg, stmname, NULL, STGM_TRANSACTED|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg2 );
1497     ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStream failed %08x\n", r);
1498 
1499     r = IStorage_OpenStream(stg, stmname, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1500     ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStream should fail %08x\n", r);
1501     if (r == S_OK)
1502         IStream_Release(stm);
1503 
1504     r = IStorage_OpenStorage(stg, stgname2, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg2 );
1505     ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStorage should fail %08x\n", r);
1506     if (r == S_OK)
1507         IStorage_Release(stg2);
1508 
1509     r = IStorage_OpenStorage(stg, stmname2, NULL, STGM_TRANSACTED|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg2 );
1510     ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStream failed %08x\n", r);
1511 
1512     r = IStorage_OpenStream(stg, stmname2, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1513     ok(r==S_OK, "IStorage->OpenStream should succeed %08x\n", r);
1514     if (r == S_OK)
1515         IStream_Release(stm);
1516 
1517     r = IStorage_OpenStorage(stg, stgname, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg2 );
1518     ok(r==S_OK, "IStorage->OpenStorage should succeed %08x\n", r);
1519     if (r == S_OK)
1520     {
1521         r = IStorage_OpenStorage(stg2, stgname, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg3 );
1522         ok(r==S_OK, "IStorage->OpenStorage should succeed %08x\n", r);
1523         if (r == S_OK)
1524             IStorage_Release(stg3);
1525 
1526         r = IStorage_OpenStorage(stg2, stgname2, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg3 );
1527         ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStorage should fail %08x\n", r);
1528         if (r == S_OK)
1529             IStorage_Release(stg3);
1530 
1531         IStorage_Release(stg2);
1532     }
1533 
1534     IStorage_Release(stg);
1535 
1536     ret = DeleteFileA(filenameA);
1537     ok(ret, "deleted file\n");
1538 }
1539 
test_substorage_share(void)1540 static void test_substorage_share(void)
1541 {
1542     IStorage *stg, *stg2, *stg3;
1543     IStream *stm, *stm2;
1544     HRESULT r;
1545     static const WCHAR stgname[] = { 'P','E','R','M','S','T','G',0 };
1546     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
1547     static const WCHAR othername[] = { 'N','E','W','N','A','M','E',0 };
1548     BOOL ret;
1549 
1550     DeleteFileA(filenameA);
1551 
1552     /* create the file */
1553     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1554                             STGM_READWRITE, 0, &stg);
1555     ok(r==S_OK, "StgCreateDocfile failed\n");
1556 
1557     /* create a read/write storage and try to open it again */
1558     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
1559     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1560 
1561     if (r == S_OK)
1562     {
1563         r = IStorage_OpenStorage(stg, stgname, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg3);
1564         ok(r==STG_E_ACCESSDENIED, "IStorage->OpenStorage should fail %08x\n", r);
1565 
1566         if (r == S_OK)
1567             IStorage_Release(stg3);
1568 
1569         r = IStorage_OpenStorage(stg, stgname, NULL, STGM_READ | STGM_SHARE_EXCLUSIVE, 0, 0, &stg3);
1570         ok(r==STG_E_ACCESSDENIED, "IStorage->OpenStorage should fail %08x\n", r);
1571 
1572         if (r == S_OK)
1573             IStorage_Release(stg3);
1574 
1575         /* cannot rename the storage while it's open */
1576         r = IStorage_RenameElement(stg, stgname, othername);
1577         ok(r==STG_E_ACCESSDENIED, "IStorage->RenameElement should fail %08x\n", r);
1578         if (SUCCEEDED(r)) IStorage_RenameElement(stg, othername, stgname);
1579 
1580         /* destroying an object while it's open invalidates it */
1581         r = IStorage_DestroyElement(stg, stgname);
1582         ok(r==S_OK, "IStorage->DestroyElement failed, hr=%08x\n", r);
1583 
1584         r = IStorage_CreateStream(stg2, stmname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stm);
1585         ok(r==STG_E_REVERTED, "IStorage->CreateStream failed, hr=%08x\n", r);
1586 
1587         if (r == S_OK)
1588             IStream_Release(stm);
1589 
1590         IStorage_Release(stg2);
1591     }
1592 
1593     /* create a read/write stream and try to open it again */
1594     r = IStorage_CreateStream(stg, stmname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stm);
1595     ok(r==S_OK, "IStorage->CreateStream failed, hr=%08x\n", r);
1596 
1597     if (r == S_OK)
1598     {
1599         r = IStorage_OpenStream(stg, stmname, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &stm2);
1600         ok(r==STG_E_ACCESSDENIED, "IStorage->OpenStream should fail %08x\n", r);
1601 
1602         if (r == S_OK)
1603             IStream_Release(stm2);
1604 
1605         r = IStorage_OpenStream(stg, stmname, NULL, STGM_READ | STGM_SHARE_EXCLUSIVE, 0, &stm2);
1606         ok(r==STG_E_ACCESSDENIED, "IStorage->OpenStream should fail %08x\n", r);
1607 
1608         if (r == S_OK)
1609             IStream_Release(stm2);
1610 
1611         /* cannot rename the stream while it's open */
1612         r = IStorage_RenameElement(stg, stmname, othername);
1613         ok(r==STG_E_ACCESSDENIED, "IStorage->RenameElement should fail %08x\n", r);
1614         if (SUCCEEDED(r)) IStorage_RenameElement(stg, othername, stmname);
1615 
1616         /* destroying an object while it's open invalidates it */
1617         r = IStorage_DestroyElement(stg, stmname);
1618         ok(r==S_OK, "IStorage->DestroyElement failed, hr=%08x\n", r);
1619 
1620         r = IStream_Write(stm, "this shouldn't work\n", 20, NULL);
1621         ok(r==STG_E_REVERTED, "IStream_Write should fail %08x\n", r);
1622 
1623         IStream_Release(stm);
1624     }
1625 
1626     IStorage_Release(stg);
1627 
1628     ret = DeleteFileA(filenameA);
1629     ok(ret, "deleted file\n");
1630 }
1631 
test_revert(void)1632 static void test_revert(void)
1633 {
1634     IStorage *stg = NULL, *stg2 = NULL, *stg3 = NULL;
1635     HRESULT r;
1636     IStream *stm = NULL, *stm2 = NULL;
1637     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
1638     static const WCHAR stmname2[] = { 'F','O','O',0 };
1639     static const WCHAR stgname[] = { 'P','E','R','M','S','T','G',0 };
1640     static const WCHAR stgname2[] = { 'T','E','M','P','S','T','G',0 };
1641     STATSTG statstg;
1642     BOOL ret;
1643 
1644     DeleteFileA(filenameA);
1645 
1646     /* create the file */
1647     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1648                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
1649     ok(r==S_OK, "StgCreateDocfile failed\n");
1650 
1651     /* commit a new stream and storage */
1652     r = IStorage_CreateStream(stg, stmname2, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1653     ok(r==S_OK, "IStorage->CreateStream failed\n");
1654 
1655     r = IStream_Write(stm, "this is stream 1\n", 16, NULL);
1656     ok(r==S_OK, "IStream->Write failed\n");
1657 
1658     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
1659     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1660 
1661     if (r == S_OK)
1662     {
1663         /* Create two substorages but only commit one */
1664         r = IStorage_CreateStorage(stg2, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg3);
1665         ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1666 
1667         if (r == S_OK)
1668             IStorage_Release(stg3);
1669 
1670         r = IStorage_Commit(stg, 0);
1671         ok(r==S_OK, "IStorage->Commit failed\n");
1672 
1673         r = IStorage_CreateStorage(stg2, stgname2, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg3);
1674         ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1675 
1676         if (r == S_OK)
1677             IStorage_Release(stg3);
1678     }
1679 
1680     /* now create a stream and storage, then revert */
1681     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm2 );
1682     ok(r==S_OK, "IStorage->CreateStream failed\n");
1683 
1684     r = IStream_Write(stm2, "this is stream 2\n", 16, NULL);
1685     ok(r==S_OK, "IStream->Write failed\n");
1686 
1687     r = IStorage_CreateStorage(stg, stgname2, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg3);
1688     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1689 
1690     r = IStorage_Revert(stg);
1691     ok(r==S_OK, "Storage_Revert failed with error 0x%08x\n", r);
1692 
1693     /* all open objects become invalid */
1694     r = IStream_Write(stm, "this shouldn't work\n", 20, NULL);
1695     ok(r==STG_E_REVERTED, "IStream_Write should fail %08x\n", r);
1696 
1697     r = IStream_Write(stm2, "this shouldn't work\n", 20, NULL);
1698     ok(r==STG_E_REVERTED, "IStream_Write should fail %08x\n", r);
1699 
1700     r = IStorage_Stat(stg2, &statstg, STATFLAG_NONAME);
1701     ok(r==STG_E_REVERTED, "IStorage_Stat should fail %08x\n", r);
1702 
1703     r = IStorage_Stat(stg3, &statstg, STATFLAG_NONAME);
1704     ok(r==STG_E_REVERTED, "IStorage_Stat should fail %08x\n", r);
1705 
1706     IStream_Release(stm);
1707     IStream_Release(stm2);
1708     IStorage_Release(stg2);
1709     IStorage_Release(stg3);
1710 
1711     r = IStorage_OpenStream(stg, stmname, NULL, STGM_SHARE_DENY_NONE|STGM_READ, 0, &stm );
1712     ok(r==STG_E_INVALIDFLAG, "IStorage->OpenStream failed %08x\n", r);
1713 
1714     r = IStorage_OpenStream(stg, stmname, NULL, STGM_DELETEONRELEASE|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1715     ok(r==STG_E_INVALIDFUNCTION, "IStorage->OpenStream failed %08x\n", r);
1716 
1717     r = IStorage_OpenStream(stg, stmname, NULL, STGM_TRANSACTED|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1718     ok(r==STG_E_INVALIDFUNCTION, "IStorage->OpenStream failed %08x\n", r);
1719 
1720     r = IStorage_OpenStorage(stg, stmname, NULL, STGM_TRANSACTED|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg2 );
1721     ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStream failed %08x\n", r);
1722 
1723     r = IStorage_OpenStream(stg, stmname, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1724     ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStream should fail %08x\n", r);
1725     if (r == S_OK)
1726         IStream_Release(stm);
1727 
1728     r = IStorage_OpenStorage(stg, stgname2, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg2 );
1729     ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStorage should fail %08x\n", r);
1730     if (r == S_OK)
1731         IStorage_Release(stg2);
1732 
1733     r = IStorage_OpenStorage(stg, stmname2, NULL, STGM_TRANSACTED|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg2 );
1734     ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStream failed %08x\n", r);
1735 
1736     r = IStorage_OpenStream(stg, stmname2, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1737     ok(r==S_OK, "IStorage->OpenStream should succeed %08x\n", r);
1738     if (r == S_OK)
1739         IStream_Release(stm);
1740 
1741     r = IStorage_OpenStorage(stg, stgname, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg2 );
1742     ok(r==S_OK, "IStorage->OpenStorage should succeed %08x\n", r);
1743     if (r == S_OK)
1744     {
1745         r = IStorage_OpenStorage(stg2, stgname, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg3 );
1746         ok(r==S_OK, "IStorage->OpenStorage should succeed %08x\n", r);
1747         if (r == S_OK)
1748             IStorage_Release(stg3);
1749 
1750         r = IStorage_OpenStorage(stg2, stgname2, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg3 );
1751         ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStorage should fail %08x\n", r);
1752         if (r == S_OK)
1753             IStorage_Release(stg3);
1754 
1755         IStorage_Release(stg2);
1756     }
1757 
1758     IStorage_Release(stg);
1759 
1760     ret = DeleteFileA(filenameA);
1761     ok(ret, "deleted file\n");
1762 
1763     /* Revert only invalidates objects in transacted mode */
1764     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1765                             STGM_READWRITE, 0, &stg);
1766     ok(r==S_OK, "StgCreateDocfile failed\n");
1767 
1768     r = IStorage_CreateStream(stg, stmname2, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1769     ok(r==S_OK, "IStorage->CreateStream failed\n");
1770 
1771     r = IStorage_Revert(stg);
1772     ok(r==S_OK, "IStorage->Revert failed %08x\n", r);
1773 
1774     r = IStream_Write(stm, "this works\n", 11, NULL);
1775     ok(r==S_OK, "IStream_Write should succeed %08x\n", r);
1776 
1777     IStream_Release(stm);
1778     IStorage_Release(stg);
1779 
1780     ret = DeleteFileA(filenameA);
1781     ok(ret, "deleted file\n");
1782 }
1783 
test_parent_free(void)1784 static void test_parent_free(void)
1785 {
1786     IStorage *stg = NULL, *stg2 = NULL, *stg3 = NULL;
1787     HRESULT r;
1788     IStream *stm = NULL;
1789     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
1790     static const WCHAR stgname[] = { 'P','E','R','M','S','T','G',0 };
1791     ULONG ref;
1792     STATSTG statstg;
1793     BOOL ret;
1794 
1795     DeleteFileA(filenameA);
1796 
1797     /* create the file */
1798     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1799                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
1800     ok(r==S_OK, "StgCreateDocfile failed\n");
1801 
1802     /* create a new storage */
1803     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
1804     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1805 
1806     if (r == S_OK)
1807     {
1808         /* now create a stream inside the new storage */
1809         r = IStorage_CreateStream(stg2, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1810         ok(r==S_OK, "IStorage->CreateStream failed\n");
1811 
1812         if (r == S_OK)
1813         {
1814             /* create a storage inside the new storage */
1815             r = IStorage_CreateStorage(stg2, stgname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stg3 );
1816             ok(r==S_OK, "IStorage->CreateStorage failed\n");
1817         }
1818 
1819         /* free the parent */
1820         ref = IStorage_Release(stg2);
1821         ok(ref == 0, "IStorage still has %u references\n", ref);
1822 
1823         /* child objects are invalid */
1824         if (r == S_OK)
1825         {
1826             r = IStream_Write(stm, "this should fail\n", 17, NULL);
1827             ok(r==STG_E_REVERTED, "IStream->Write should fail, hr=%x\n", r);
1828 
1829             IStream_Release(stm);
1830 
1831             r = IStorage_Stat(stg3, &statstg, STATFLAG_NONAME);
1832             ok(r==STG_E_REVERTED, "IStorage_Stat should fail %08x\n", r);
1833 
1834             r = IStorage_SetStateBits(stg3, 1, 1);
1835             ok(r==STG_E_REVERTED, "IStorage_Stat should fail %08x\n", r);
1836 
1837             IStorage_Release(stg3);
1838         }
1839     }
1840 
1841     IStorage_Release(stg);
1842 
1843     ret = DeleteFileA(filenameA);
1844     ok(ret, "deleted file\n");
1845 }
1846 
test_nonroot_transacted(void)1847 static void test_nonroot_transacted(void)
1848 {
1849     IStorage *stg = NULL, *stg2 = NULL, *stg3 = NULL;
1850     HRESULT r;
1851     IStream *stm = NULL;
1852     static const WCHAR stgname[] = { 'P','E','R','M','S','T','G',0 };
1853     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
1854     static const WCHAR stmname2[] = { 'F','O','O',0 };
1855     BOOL ret;
1856 
1857     DeleteFileA(filenameA);
1858 
1859     /* create a transacted file */
1860     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1861                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
1862     ok(r==S_OK, "StgCreateDocfile failed\n");
1863 
1864     /* create a transacted substorage */
1865     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE | STGM_TRANSACTED, 0, 0, &stg2);
1866     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1867 
1868     if (r == S_OK)
1869     {
1870         /* create and commit stmname */
1871         r = IStorage_CreateStream(stg2, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1872         ok(r==S_OK, "IStorage->CreateStream failed\n");
1873         if (r == S_OK)
1874             IStream_Release(stm);
1875 
1876         IStorage_Commit(stg2, 0);
1877 
1878         /* create and revert stmname2 */
1879         r = IStorage_CreateStream(stg2, stmname2, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1880         ok(r==S_OK, "IStorage->CreateStream failed\n");
1881         if (r == S_OK)
1882             IStream_Release(stm);
1883 
1884         IStorage_Revert(stg2);
1885 
1886         /* check that Commit and Revert really worked */
1887         r = IStorage_OpenStream(stg2, stmname, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1888         ok(r==S_OK, "IStorage->OpenStream should succeed %08x\n", r);
1889         if (r == S_OK)
1890             IStream_Release(stm);
1891 
1892         r = IStorage_OpenStream(stg2, stmname2, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1893         ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStream should fail %08x\n", r);
1894         if (r == S_OK)
1895             IStream_Release(stm);
1896 
1897         IStorage_Release(stg2);
1898     }
1899 
1900     /* create a read-only transacted substorage */
1901     r = IStorage_OpenStorage(stg, stgname, NULL, STGM_READ | STGM_SHARE_EXCLUSIVE | STGM_TRANSACTED, NULL, 0, &stg2);
1902     ok(r==S_OK, "IStorage->OpenStorage failed, hr=%08x\n", r);
1903 
1904     if (r == S_OK)
1905     {
1906         /* The storage can be modified. */
1907         r = IStorage_CreateStorage(stg2, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg3);
1908         ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1909         if (r == S_OK)
1910             IStorage_Release(stg3);
1911 
1912         /* But changes cannot be committed. */
1913         r = IStorage_Commit(stg2, 0);
1914         ok(r==STG_E_ACCESSDENIED, "IStorage->Commit should fail, hr=%08x\n", r);
1915 
1916         IStorage_Release(stg2);
1917     }
1918 
1919     IStorage_Release(stg);
1920 
1921     /* create a non-transacted file */
1922     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
1923                             STGM_READWRITE, 0, &stg);
1924     ok(r==S_OK, "StgCreateDocfile failed\n");
1925 
1926     /* create a transacted substorage */
1927     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE | STGM_TRANSACTED, 0, 0, &stg2);
1928     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
1929 
1930     if (r == S_OK)
1931     {
1932         /* create and commit stmname */
1933         r = IStorage_CreateStream(stg2, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1934         ok(r==S_OK, "IStorage->CreateStream failed\n");
1935         if (r == S_OK)
1936             IStream_Release(stm);
1937 
1938         IStorage_Commit(stg2, 0);
1939 
1940         /* create and revert stmname2 */
1941         r = IStorage_CreateStream(stg2, stmname2, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm );
1942         ok(r==S_OK, "IStorage->CreateStream failed\n");
1943         if (r == S_OK)
1944             IStream_Release(stm);
1945 
1946         IStorage_Revert(stg2);
1947 
1948         /* check that Commit and Revert really worked */
1949         r = IStorage_OpenStream(stg2, stmname, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1950         ok(r==S_OK, "IStorage->OpenStream should succeed %08x\n", r);
1951         if (r == S_OK)
1952             IStream_Release(stm);
1953 
1954         r = IStorage_OpenStream(stg2, stmname2, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, &stm );
1955         ok(r==STG_E_FILENOTFOUND, "IStorage->OpenStream should fail %08x\n", r);
1956         if (r == S_OK)
1957             IStream_Release(stm);
1958 
1959         IStorage_Release(stg2);
1960     }
1961 
1962     IStorage_Release(stg);
1963 
1964     ret = DeleteFileA(filenameA);
1965     ok(ret, "deleted file\n");
1966 }
1967 
test_ReadClassStm(void)1968 static void test_ReadClassStm(void)
1969 {
1970     CLSID clsid, clsid2;
1971     HRESULT hr;
1972     IStream *pStream;
1973     static const LARGE_INTEGER llZero;
1974 
1975     hr = ReadClassStm(NULL, &clsid);
1976     ok(hr == E_INVALIDARG, "ReadClassStm should have returned E_INVALIDARG instead of 0x%08x\n", hr);
1977 
1978     hr = CreateStreamOnHGlobal(NULL, TRUE, &pStream);
1979     ok_ole_success(hr, "CreateStreamOnHGlobal");
1980     hr = WriteClassStm(pStream, &test_stg_cls);
1981     ok_ole_success(hr, "WriteClassStm");
1982 
1983     hr = ReadClassStm(pStream, NULL);
1984     ok(hr == E_INVALIDARG, "ReadClassStm should have returned E_INVALIDARG instead of 0x%08x\n", hr);
1985 
1986     memset(&clsid, 0xcc, sizeof(clsid));
1987     memset(&clsid2, 0xcc, sizeof(clsid2));
1988     hr = ReadClassStm(NULL, &clsid);
1989     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
1990     ok(IsEqualCLSID(&clsid, &clsid2), "got wrong clsid\n");
1991 
1992     /* test not rewound stream */
1993     hr = ReadClassStm(pStream, &clsid);
1994     ok(hr == STG_E_READFAULT, "ReadClassStm should have returned STG_E_READFAULT instead of 0x%08x\n", hr);
1995     ok(IsEqualCLSID(&clsid, &CLSID_NULL), "clsid should have been zeroed\n");
1996 
1997     hr = IStream_Seek(pStream, llZero, STREAM_SEEK_SET, NULL);
1998     ok_ole_success(hr, "IStream_Seek");
1999     hr = ReadClassStm(pStream, &clsid);
2000     ok_ole_success(hr, "ReadClassStm");
2001     ok(IsEqualCLSID(&clsid, &test_stg_cls), "clsid should have been set to CLSID_WineTest\n");
2002 
2003     IStream_Release(pStream);
2004 }
2005 
2006 struct access_res
2007 {
2008     BOOL gothandle;
2009     DWORD lasterr;
2010     BOOL ignore;
2011 };
2012 
2013 static const struct access_res create[16] =
2014 {
2015     { TRUE, ERROR_SUCCESS, TRUE },
2016     { TRUE, ERROR_SUCCESS, TRUE },
2017     { TRUE, ERROR_SUCCESS, FALSE },
2018     { TRUE, ERROR_SUCCESS, FALSE },
2019     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2020     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2021     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2022     { TRUE, ERROR_SUCCESS, FALSE },
2023     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2024     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2025     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2026     { TRUE, ERROR_SUCCESS, TRUE },
2027     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2028     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2029     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2030     { TRUE, ERROR_SUCCESS, TRUE }
2031 };
2032 
2033 static const struct access_res create_commit[16] =
2034 {
2035     { TRUE, ERROR_SUCCESS, TRUE },
2036     { TRUE, ERROR_SUCCESS, TRUE },
2037     { TRUE, ERROR_SUCCESS, FALSE },
2038     { TRUE, ERROR_SUCCESS, FALSE },
2039     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2040     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2041     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2042     { TRUE, ERROR_SUCCESS, FALSE },
2043     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2044     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2045     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2046     { TRUE, ERROR_SUCCESS, TRUE },
2047     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2048     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2049     { FALSE, ERROR_SHARING_VIOLATION, FALSE },
2050     { TRUE, ERROR_SUCCESS, TRUE }
2051 };
2052 
2053 static const struct access_res create_close[16] =
2054 {
2055     { TRUE, ERROR_SUCCESS, FALSE },
2056     { TRUE, ERROR_SUCCESS, FALSE },
2057     { TRUE, ERROR_SUCCESS, FALSE },
2058     { TRUE, ERROR_SUCCESS, FALSE },
2059     { TRUE, ERROR_SUCCESS, FALSE },
2060     { TRUE, ERROR_SUCCESS, FALSE },
2061     { TRUE, ERROR_SUCCESS, FALSE },
2062     { TRUE, ERROR_SUCCESS, FALSE },
2063     { TRUE, ERROR_SUCCESS, FALSE },
2064     { TRUE, ERROR_SUCCESS, FALSE },
2065     { TRUE, ERROR_SUCCESS, FALSE },
2066     { TRUE, ERROR_SUCCESS, FALSE },
2067     { TRUE, ERROR_SUCCESS, FALSE },
2068     { TRUE, ERROR_SUCCESS, FALSE },
2069     { TRUE, ERROR_SUCCESS, FALSE },
2070     { TRUE, ERROR_SUCCESS }
2071 };
2072 
2073 static const DWORD access_modes[4] = {
2074     0,
2075     GENERIC_READ,
2076     GENERIC_WRITE,
2077     GENERIC_READ | GENERIC_WRITE
2078 };
2079 
2080 static const DWORD share_modes[4] = {
2081     0,
2082     FILE_SHARE_READ,
2083     FILE_SHARE_WRITE,
2084     FILE_SHARE_READ | FILE_SHARE_WRITE
2085 };
2086 
_test_file_access(LPCSTR file,const struct access_res * ares,DWORD line)2087 static void _test_file_access(LPCSTR file, const struct access_res *ares, DWORD line)
2088 {
2089     int i, j, idx = 0;
2090 
2091     for (i = 0; i < ARRAY_SIZE(access_modes); i++)
2092     {
2093         for (j = 0; j < ARRAY_SIZE(share_modes); j++)
2094         {
2095             DWORD lasterr;
2096             HANDLE hfile;
2097 
2098             if (ares[idx].ignore)
2099                 continue;
2100 
2101             SetLastError(0xdeadbeef);
2102             hfile = CreateFileA(file, access_modes[i], share_modes[j], NULL, OPEN_EXISTING,
2103                                 FILE_ATTRIBUTE_NORMAL, 0);
2104             lasterr = GetLastError();
2105 
2106             ok((hfile != INVALID_HANDLE_VALUE) == ares[idx].gothandle,
2107                "(%d, handle, %d): Expected %d, got %d\n",
2108                line, idx, ares[idx].gothandle,
2109                (hfile != INVALID_HANDLE_VALUE));
2110 
2111             ok(lasterr == ares[idx].lasterr ||
2112                broken(lasterr == 0xdeadbeef) /* win9x */,
2113                "(%d, lasterr, %d): Expected %d, got %d\n",
2114                line, idx, ares[idx].lasterr, lasterr);
2115 
2116             CloseHandle(hfile);
2117             idx++;
2118         }
2119     }
2120 }
2121 
2122 #define test_file_access(file, ares) _test_file_access(file, ares, __LINE__)
2123 
test_access(void)2124 static void test_access(void)
2125 {
2126     static const WCHAR fileW[] = {'w','i','n','e','t','e','s','t',0};
2127     static const char fileA[] = "winetest";
2128     IStorage *stg;
2129     HRESULT hr;
2130 
2131     /* STGM_TRANSACTED */
2132     hr = StgCreateDocfile(fileW, STGM_CREATE | STGM_READWRITE |
2133                           STGM_SHARE_EXCLUSIVE | STGM_TRANSACTED, 0, &stg);
2134     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2135 
2136     test_file_access(fileA, create);
2137 
2138     hr = IStorage_Commit(stg, STGC_DEFAULT);
2139     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2140 
2141     test_file_access(fileA, create_commit);
2142 
2143     IStorage_Release(stg);
2144 
2145     test_file_access(fileA, create_close);
2146 
2147     DeleteFileA(fileA);
2148 
2149     /* STGM_DIRECT */
2150     hr = StgCreateDocfile(fileW, STGM_CREATE | STGM_READWRITE |
2151                           STGM_SHARE_EXCLUSIVE | STGM_DIRECT, 0, &stg);
2152     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2153 
2154     test_file_access(fileA, create);
2155 
2156     hr = IStorage_Commit(stg, STGC_DEFAULT);
2157     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2158 
2159     test_file_access(fileA, create_commit);
2160 
2161     IStorage_Release(stg);
2162 
2163     test_file_access(fileA, create_close);
2164 
2165     DeleteFileA(fileA);
2166 
2167     /* STGM_SHARE_DENY_NONE */
2168     hr = StgCreateDocfile(fileW, STGM_CREATE | STGM_READWRITE |
2169                           STGM_SHARE_DENY_NONE | STGM_TRANSACTED, 0, &stg);
2170     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2171 
2172     test_file_access(fileA, create);
2173 
2174     hr = IStorage_Commit(stg, STGC_DEFAULT);
2175     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2176 
2177     test_file_access(fileA, create_commit);
2178 
2179     IStorage_Release(stg);
2180 
2181     test_file_access(fileA, create_close);
2182 
2183     DeleteFileA(fileA);
2184 
2185     /* STGM_SHARE_DENY_READ */
2186     hr = StgCreateDocfile(fileW, STGM_CREATE | STGM_READWRITE |
2187                           STGM_SHARE_DENY_READ | STGM_TRANSACTED, 0, &stg);
2188     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2189 
2190     test_file_access(fileA, create);
2191 
2192     hr = IStorage_Commit(stg, STGC_DEFAULT);
2193     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2194 
2195     test_file_access(fileA, create_commit);
2196 
2197     IStorage_Release(stg);
2198 
2199     test_file_access(fileA, create_close);
2200 
2201     DeleteFileA(fileA);
2202 
2203     /* STGM_SHARE_DENY_WRITE */
2204     hr = StgCreateDocfile(fileW, STGM_CREATE | STGM_READWRITE |
2205                           STGM_SHARE_DENY_WRITE | STGM_TRANSACTED, 0, &stg);
2206     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2207 
2208     test_file_access(fileA, create);
2209 
2210     hr = IStorage_Commit(stg, STGC_DEFAULT);
2211     ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2212 
2213     test_file_access(fileA, create_commit);
2214 
2215     IStorage_Release(stg);
2216 
2217     test_file_access(fileA, create_close);
2218 
2219     DeleteFileA(fileA);
2220 
2221     /* STGM_DIRECT_SWMR | STGM_READ | STGM_SHARE_DENY_NONE - reader mode for direct SWMR mode */
2222     hr = StgCreateDocfile(fileW, STGM_CREATE | STGM_READWRITE | STGM_SHARE_DENY_WRITE | STGM_TRANSACTED, 0, &stg);
2223     ok(hr == S_OK, "got %08x\n", hr);
2224     IStorage_Release(stg);
2225 
2226     hr = StgOpenStorage(fileW, NULL, STGM_DIRECT_SWMR | STGM_READ | STGM_SHARE_DENY_NONE, NULL, 0, &stg);
2227     ok(hr == S_OK || broken(hr == STG_E_INVALIDFLAG), "got %08x\n", hr);
2228     if(hr != S_OK)
2229        return;
2230 
2231     test_file_access(fileA, create);
2232 
2233     IStorage_Release(stg);
2234     test_file_access(fileA, create_close);
2235 
2236     DeleteFileA(fileA);
2237 }
2238 
test_readonly(void)2239 static void test_readonly(void)
2240 {
2241     IStorage *stg, *stg2, *stg3;
2242     IStream *stream;
2243     HRESULT hr;
2244     static const WCHAR fileW[] = {'w','i','n','e','t','e','s','t',0};
2245     static const WCHAR storageW[] = {'s','t','o','r','a','g','e',0};
2246     static const WCHAR streamW[] = {'s','t','r','e','a','m',0};
2247 
2248     hr = StgCreateDocfile( fileW, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stg);
2249     ok(hr == S_OK, "should succeed, res=%x\n", hr);
2250     if (SUCCEEDED(hr))
2251     {
2252         hr = IStorage_CreateStorage( stg, storageW, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stg2 );
2253         ok(hr == S_OK, "should succeed, res=%x\n", hr);
2254         if (SUCCEEDED(hr))
2255         {
2256             hr = IStorage_CreateStream( stg2, streamW, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stream );
2257             ok(hr == S_OK, "should succeed, res=%x\n", hr);
2258             if (SUCCEEDED(hr))
2259                 IStream_Release(stream);
2260             IStorage_Release(stg2);
2261         }
2262         IStorage_Release(stg);
2263     }
2264 
2265     /* re-open read only */
2266     hr = StgOpenStorage( fileW, NULL, STGM_TRANSACTED | STGM_SHARE_DENY_NONE | STGM_READ, NULL, 0, &stg);
2267     ok(hr == S_OK, "should succeed, res=%x\n", hr);
2268     if (SUCCEEDED(hr))
2269     {
2270         hr = IStorage_OpenStorage( stg, storageW, NULL, STGM_SHARE_EXCLUSIVE | STGM_READ, NULL, 0, &stg2 );
2271         ok(hr == S_OK, "should succeed, res=%x\n", hr);
2272         if (SUCCEEDED(hr))
2273         {
2274             /* CreateStream on read-only storage, name exists */
2275             hr = IStorage_CreateStream( stg2, streamW, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READ, 0, 0, &stream );
2276             ok(hr == STG_E_ACCESSDENIED, "should fail, res=%x\n", hr);
2277             if (SUCCEEDED(hr))
2278                 IStream_Release(stream);
2279 
2280             /* CreateStream on read-only storage, name does not exist */
2281             hr = IStorage_CreateStream( stg2, storageW, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READ, 0, 0, &stream );
2282             ok(hr == STG_E_ACCESSDENIED, "should fail, res=%x\n", hr);
2283             if (SUCCEEDED(hr))
2284                 IStream_Release(stream);
2285 
2286             /* CreateStorage on read-only storage, name exists */
2287             hr = IStorage_CreateStorage( stg2, streamW, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READ, 0, 0, &stg3 );
2288             ok(hr == STG_E_FILEALREADYEXISTS, "should fail, res=%x\n", hr);
2289             if (SUCCEEDED(hr))
2290                 IStorage_Release(stg3);
2291 
2292             /* CreateStorage on read-only storage, name does not exist */
2293             hr = IStorage_CreateStorage( stg2, storageW, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READ, 0, 0, &stg3 );
2294             ok(hr == STG_E_ACCESSDENIED, "should fail, res=%x\n", hr);
2295             if (SUCCEEDED(hr))
2296                 IStorage_Release(stg3);
2297 
2298             /* DestroyElement on read-only storage, name exists */
2299             hr = IStorage_DestroyElement( stg2, streamW );
2300             ok(hr == STG_E_ACCESSDENIED, "should fail, res=%x\n", hr);
2301 
2302             /* DestroyElement on read-only storage, name does not exist */
2303             hr = IStorage_DestroyElement( stg2, storageW );
2304             ok(hr == STG_E_ACCESSDENIED, "should fail, res=%x\n", hr);
2305 
2306             IStorage_Release(stg2);
2307         }
2308 
2309         IStorage_Release(stg);
2310     }
2311 
2312     DeleteFileA("winetest");
2313 }
2314 
test_simple(void)2315 static void test_simple(void)
2316 {
2317     /* Tests for STGM_SIMPLE mode */
2318 
2319     IStorage *stg, *stg2;
2320     HRESULT r;
2321     IStream *stm;
2322     static const WCHAR stgname[] = { 'S','t','g',0 };
2323     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
2324     static const WCHAR stmname2[] = { 'S','m','a','l','l',0 };
2325     LARGE_INTEGER pos;
2326     ULARGE_INTEGER upos;
2327     DWORD count;
2328     STATSTG stat;
2329 
2330     DeleteFileA(filenameA);
2331 
2332     r = StgCreateDocfile( filename, STGM_SIMPLE | STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stg);
2333     ok(r == S_OK, "got %08x\n", r);
2334 
2335     r = IStorage_CreateStorage(stg, stgname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stg2);
2336     ok(r == STG_E_INVALIDFUNCTION, "got %08x\n", r);
2337     if (SUCCEEDED(r)) IStorage_Release(stg2);
2338 
2339     r = IStorage_CreateStream(stg, stmname, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm);
2340     ok(r == STG_E_INVALIDFLAG, "got %08x\n", r);
2341     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm);
2342     ok(r == S_OK, "got %08x\n", r);
2343 
2344     upos.QuadPart = 6000;
2345     r = IStream_SetSize(stm, upos);
2346     ok(r == S_OK, "got %08x\n", r);
2347 
2348     r = IStream_Write(stm, "foo", 3, &count);
2349     ok(r == S_OK, "got %08x\n", r);
2350     ok(count == 3, "got %d\n", count);
2351 
2352     pos.QuadPart = 0;
2353     r = IStream_Seek(stm, pos, STREAM_SEEK_CUR, &upos);
2354     ok(r == S_OK, "got %08x\n", r);
2355     ok(upos.QuadPart == 3, "got %d\n", upos.u.LowPart);
2356 
2357     r = IStream_Stat(stm, &stat, STATFLAG_NONAME);
2358     ok(r == S_OK ||
2359        broken(r == STG_E_INVALIDFUNCTION), /* NT4 and below */
2360        "got %08x\n", r);
2361     if (r == S_OK)
2362         ok(stat.cbSize.QuadPart == 3, "got %d\n", stat.cbSize.u.LowPart);
2363 
2364     pos.QuadPart = 1;
2365     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &upos);
2366     ok(r == S_OK, "got %08x\n", r);
2367     ok(upos.QuadPart == 1, "got %d\n", upos.u.LowPart);
2368 
2369     r = IStream_Stat(stm, &stat, STATFLAG_NONAME);
2370     ok(r == S_OK ||
2371        broken(r == STG_E_INVALIDFUNCTION), /* NT4 and below */
2372        "got %08x\n", r);
2373     if (r == S_OK)
2374         ok(stat.cbSize.QuadPart == 1, "got %d\n", stat.cbSize.u.LowPart);
2375 
2376     IStream_Release(stm);
2377 
2378     r = IStorage_CreateStream(stg, stmname2, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm);
2379     ok(r == S_OK, "got %08x\n", r);
2380 
2381     upos.QuadPart = 100;
2382     r = IStream_SetSize(stm, upos);
2383     ok(r == S_OK, "got %08x\n", r);
2384 
2385     r = IStream_Write(stm, "foo", 3, &count);
2386     ok(r == S_OK, "got %08x\n", r);
2387     ok(count == 3, "got %d\n", count);
2388 
2389     IStream_Release(stm);
2390 
2391     IStorage_Commit(stg, STGC_DEFAULT);
2392     IStorage_Release(stg);
2393 
2394     r = StgOpenStorage( filename, NULL, STGM_SIMPLE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, NULL, 0, &stg);
2395     if (r == STG_E_INVALIDFLAG)
2396     {
2397         win_skip("Flag combination is not supported on NT4 and below\n");
2398         DeleteFileA(filenameA);
2399         return;
2400     }
2401     ok(r == S_OK, "got %08x\n", r);
2402 
2403     r = IStorage_OpenStorage(stg, stgname, NULL, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, NULL, 0, &stg2);
2404     ok(r == STG_E_INVALIDFUNCTION, "got %08x\n", r);
2405     if (SUCCEEDED(r)) IStorage_Release(stg2);
2406 
2407     r = IStorage_OpenStream(stg, stmname, NULL, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stm);
2408     ok(r == S_OK, "got %08x\n", r);
2409 
2410     r = IStream_Stat(stm, &stat, STATFLAG_NONAME);
2411     ok(r == S_OK, "got %08x\n", r);
2412     ok(stat.cbSize.QuadPart == 6000, "got %d\n", stat.cbSize.u.LowPart);
2413 
2414     IStream_Release(stm);
2415 
2416     r = IStorage_OpenStream(stg, stmname2, NULL, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stm);
2417     ok(r == S_OK, "got %08x\n", r);
2418 
2419     r = IStream_Stat(stm, &stat, STATFLAG_NONAME);
2420     ok(r == S_OK, "got %08x\n", r);
2421     ok(stat.cbSize.QuadPart == 4096, "got %d\n", stat.cbSize.u.LowPart);
2422 
2423     IStream_Release(stm);
2424 
2425 
2426     IStorage_Release(stg);
2427 
2428     DeleteFileA(filenameA);
2429 }
2430 
test_fmtusertypestg(void)2431 static void test_fmtusertypestg(void)
2432 {
2433     IStorage *stg;
2434     IEnumSTATSTG *stat;
2435     HRESULT hr;
2436     static const char fileA[]  = {'f','m','t','t','e','s','t',0};
2437     static const WCHAR fileW[] = {'f','m','t','t','e','s','t',0};
2438     static WCHAR userTypeW[] = {'S','t','g','U','s','r','T','y','p','e',0};
2439     static const WCHAR strmNameW[] = {1,'C','o','m','p','O','b','j',0};
2440     static const STATSTG statstg_null;
2441 
2442     hr = StgCreateDocfile( fileW, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stg);
2443     ok(hr == S_OK, "should succeed, res=%x\n", hr);
2444 
2445     if (SUCCEEDED(hr))
2446     {
2447         /* try to write the stream */
2448         hr = WriteFmtUserTypeStg(stg, 0, userTypeW);
2449         ok(hr == S_OK, "should succeed, res=%x\n", hr);
2450 
2451         /* check that the stream was created */
2452         hr = IStorage_EnumElements(stg, 0, NULL, 0, &stat);
2453         ok(hr == S_OK, "should succeed, res=%x\n", hr);
2454         if (SUCCEEDED(hr))
2455         {
2456             BOOL found = FALSE;
2457             STATSTG statstg;
2458             DWORD got;
2459             memset(&statstg, 0xad, sizeof(statstg));
2460             while ((hr = IEnumSTATSTG_Next(stat, 1, &statstg, &got)) == S_OK && got == 1)
2461             {
2462                 if (strcmp_ww(statstg.pwcsName, strmNameW) == 0)
2463                     found = TRUE;
2464                 else
2465                     ok(0, "found unexpected stream or storage\n");
2466                 CoTaskMemFree(statstg.pwcsName);
2467             }
2468             ok(memcmp(&statstg, &statstg_null, sizeof(statstg)) == 0, "statstg is not zeroed\n");
2469             ok(found == TRUE, "expected storage to contain stream \\0001CompObj\n");
2470             IEnumSTATSTG_Release(stat);
2471         }
2472 
2473         /* re-write the stream */
2474         hr = WriteFmtUserTypeStg(stg, 0, userTypeW);
2475         ok(hr == S_OK, "should succeed, res=%x\n", hr);
2476 
2477         /* check that the stream is still there */
2478         hr = IStorage_EnumElements(stg, 0, NULL, 0, &stat);
2479         ok(hr == S_OK, "should succeed, res=%x\n", hr);
2480         if (SUCCEEDED(hr))
2481         {
2482             BOOL found = FALSE;
2483             STATSTG statstg;
2484             DWORD got;
2485             memset(&statstg, 0xad, sizeof(statstg));
2486             while ((hr = IEnumSTATSTG_Next(stat, 1, &statstg, &got)) == S_OK && got == 1)
2487             {
2488                 if (strcmp_ww(statstg.pwcsName, strmNameW) == 0)
2489                     found = TRUE;
2490                 else
2491                     ok(0, "found unexpected stream or storage\n");
2492                 CoTaskMemFree(statstg.pwcsName);
2493             }
2494             ok(memcmp(&statstg, &statstg_null, sizeof(statstg)) == 0, "statstg is not zeroed\n");
2495             ok(found == TRUE, "expected storage to contain stream \\0001CompObj\n");
2496             IEnumSTATSTG_Release(stat);
2497         }
2498 
2499         IStorage_Release(stg);
2500         DeleteFileA( fileA );
2501     }
2502 }
2503 
test_references(void)2504 static void test_references(void)
2505 {
2506     IStorage *stg,*stg2;
2507     HRESULT hr;
2508     unsigned c1,c2;
2509     static const WCHAR StorName[] = { 'D','a','t','a','S','p','a','c','e','I','n','f','o',0 };
2510 
2511     DeleteFileA(filenameA);
2512 
2513     hr = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
2514     ok(hr==S_OK, "StgCreateDocfile failed\n");
2515 
2516     if (SUCCEEDED(hr))
2517     {
2518         IStorage_Release(stg);
2519 
2520         hr = StgOpenStorage( filename, NULL, STGM_TRANSACTED | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, NULL, 0, &stg);
2521         ok(hr==S_OK, "StgOpenStorage failed (result=%x)\n",hr);
2522 
2523         if (SUCCEEDED(hr))
2524         {
2525             hr = IStorage_CreateStorage(stg,StorName,STGM_READWRITE | STGM_SHARE_EXCLUSIVE,0,0,&stg2);
2526             ok(hr == S_OK, "IStorage_CreateStorage failed (result=%x)\n",hr);
2527 
2528             if (SUCCEEDED(hr))
2529             {
2530                 c1 = IStorage_AddRef(stg);
2531                 ok(c1 == 2, "creating internal storage added references to ancestor\n");
2532                 c1 = IStorage_AddRef(stg);
2533                 IStorage_Release(stg2);
2534                 c2 = IStorage_AddRef(stg) - 1;
2535                 ok(c1 == c2, "releasing internal storage removed references to ancestor\n");
2536             }
2537             c1 = IStorage_Release(stg);
2538             while ( c1 ) c1 = IStorage_Release(stg);
2539         }
2540     }
2541 
2542     DeleteFileA(filenameA);
2543 }
2544 
2545 /* dest
2546  *  |-StorageA
2547  *  |  `StreamA: "StreamA"
2548  *  |-StorageB
2549  *  |  `StreamB: "StreamB"
2550  *  `StreamC: "StreamC"
2551  */
create_test_file(IStorage * dest)2552 static HRESULT create_test_file(IStorage *dest)
2553 {
2554     IStorage *stgA = NULL, *stgB = NULL;
2555     IStream *strmA = NULL, *strmB = NULL, *strmC = NULL;
2556     const ULONG strmA_name_size = lstrlenW(strmA_name) * sizeof(WCHAR);
2557     const ULONG strmB_name_size = lstrlenW(strmB_name) * sizeof(WCHAR);
2558     const ULONG strmC_name_size = lstrlenW(strmC_name) * sizeof(WCHAR);
2559     ULONG bytes;
2560     HRESULT hr;
2561 
2562     hr = IStorage_CreateStorage(dest, stgA_name, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stgA);
2563     ok(hr == S_OK, "IStorage_CreateStorage failed: 0x%08x\n", hr);
2564     if(FAILED(hr))
2565         goto cleanup;
2566 
2567     hr = IStorage_CreateStream(stgA, strmA_name, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &strmA);
2568     ok(hr == S_OK, "IStorage_CreateStream failed: 0x%08x\n", hr);
2569     if(FAILED(hr))
2570         goto cleanup;
2571 
2572     hr = IStream_Write(strmA, strmA_name, strmA_name_size, &bytes);
2573     ok(hr == S_OK && bytes == strmA_name_size, "IStream_Write failed: 0x%08x, %d of %d bytes written\n", hr, bytes, strmA_name_size);
2574 
2575     hr = IStorage_CreateStorage(dest, stgB_name, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stgB);
2576     ok(hr == S_OK, "IStorage_CreateStorage failed: 0x%08x\n", hr);
2577     if(FAILED(hr))
2578         goto cleanup;
2579 
2580     hr = IStorage_CreateStream(stgB, strmB_name, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &strmB);
2581     ok(hr == S_OK, "IStorage_CreateStream failed: 0x%08x\n", hr);
2582     if(FAILED(hr))
2583         goto cleanup;
2584 
2585     hr = IStream_Write(strmB, strmB_name, strmB_name_size, &bytes);
2586     ok(hr == S_OK && bytes == strmB_name_size, "IStream_Write failed: 0x%08x, %d of %d bytes written\n", hr, bytes, strmB_name_size);
2587 
2588     hr = IStorage_CreateStream(dest, strmC_name, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &strmC);
2589     ok(hr == S_OK, "IStorage_CreateStream failed: 0x%08x\n", hr);
2590     if(FAILED(hr))
2591         goto cleanup;
2592 
2593     hr = IStream_Write(strmC, strmC_name, strmC_name_size, &bytes);
2594     ok(hr == S_OK && bytes == strmC_name_size, "IStream_Write failed: 0x%08x, %d of %d bytes written\n", hr, bytes, strmC_name_size);
2595 
2596 cleanup:
2597     if(strmC)
2598         IStream_Release(strmC);
2599     if(strmB)
2600         IStream_Release(strmB);
2601     if(stgB)
2602         IStorage_Release(stgB);
2603     if(strmA)
2604         IStream_Release(strmA);
2605     if(stgA)
2606         IStorage_Release(stgA);
2607 
2608     return hr;
2609 }
2610 
test_copyto(void)2611 static void test_copyto(void)
2612 {
2613     IStorage *file1 = NULL, *file2 = NULL, *stg_tmp;
2614     IStream *strm_tmp;
2615     WCHAR buf[64];
2616     HRESULT hr;
2617 
2618     /* create & populate file1 */
2619     hr = StgCreateDocfile(file1_name, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &file1);
2620     ok(hr == S_OK, "StgCreateDocfile failed: 0x%08x\n", hr);
2621     if(FAILED(hr))
2622         goto cleanup;
2623 
2624     hr = create_test_file(file1);
2625     if(FAILED(hr))
2626         goto cleanup;
2627 
2628     /* create file2 */
2629     hr = StgCreateDocfile(file2_name, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &file2);
2630     ok(hr == S_OK, "StgCreateDocfile failed: 0x%08x\n", hr);
2631     if(FAILED(hr))
2632         goto cleanup;
2633 
2634     /* copy file1 into file2 */
2635     hr = IStorage_CopyTo(file1, 0, NULL, NULL, NULL);
2636     ok(hr == STG_E_INVALIDPOINTER, "CopyTo should give STG_E_INVALIDPONITER, gave: 0x%08x\n", hr);
2637 
2638     hr = IStorage_CopyTo(file1, 0, NULL, NULL, file2);
2639     ok(hr == S_OK, "CopyTo failed: 0x%08x\n", hr);
2640     if(FAILED(hr))
2641         goto cleanup;
2642 
2643     /* verify that all of file1 was copied */
2644     hr = IStorage_OpenStorage(file2, stgA_name, NULL,
2645             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg_tmp);
2646     ok(hr == S_OK, "OpenStorage failed: 0x%08x\n", hr);
2647 
2648     if(SUCCEEDED(hr)){
2649         hr = IStorage_OpenStream(stg_tmp, strmA_name, NULL,
2650                 STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2651         ok(hr == S_OK, "OpenStream failed: 0x%08x\n", hr);
2652 
2653         if(SUCCEEDED(hr)){
2654             memset(buf, 0, sizeof(buf));
2655             hr = IStream_Read(strm_tmp, buf, sizeof(buf), NULL);
2656             ok(hr == S_OK, "Read failed: 0x%08x\n", hr);
2657             if(SUCCEEDED(hr))
2658                 ok(strcmp_ww(buf, strmA_name) == 0,
2659                         "Expected %s to be read, got %s\n", wine_dbgstr_w(strmA_name), wine_dbgstr_w(buf));
2660 
2661             IStream_Release(strm_tmp);
2662         }
2663 
2664         IStorage_Release(stg_tmp);
2665     }
2666 
2667     hr = IStorage_OpenStorage(file2, stgB_name, NULL,
2668             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg_tmp);
2669     ok(hr == S_OK, "OpenStorage failed: 0x%08x\n", hr);
2670 
2671     if(SUCCEEDED(hr)){
2672         hr = IStorage_OpenStream(stg_tmp, strmB_name, NULL,
2673                 STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2674         ok(hr == S_OK, "OpenStream failed: 0x%08x\n", hr);
2675 
2676         if(SUCCEEDED(hr)){
2677             memset(buf, 0, sizeof(buf));
2678             hr = IStream_Read(strm_tmp, buf, sizeof(buf), NULL);
2679             ok(hr == S_OK, "Read failed: 0x%08x\n", hr);
2680             if(SUCCEEDED(hr))
2681                 ok(strcmp_ww(buf, strmB_name) == 0,
2682                         "Expected %s to be read, got %s\n", wine_dbgstr_w(strmB_name), wine_dbgstr_w(buf));
2683 
2684             IStream_Release(strm_tmp);
2685         }
2686 
2687         IStorage_Release(stg_tmp);
2688     }
2689 
2690     hr = IStorage_OpenStream(file2, strmC_name, NULL,
2691             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2692     ok(hr == S_OK, "OpenStream failed: 0x%08x\n", hr);
2693 
2694     if(SUCCEEDED(hr)){
2695         memset(buf, 0, sizeof(buf));
2696         hr = IStream_Read(strm_tmp, buf, sizeof(buf), NULL);
2697         ok(hr == S_OK, "Read failed: 0x%08x\n", hr);
2698         if(SUCCEEDED(hr))
2699             ok(strcmp_ww(buf, strmC_name) == 0,
2700                     "Expected %s to be read, got %s\n", wine_dbgstr_w(strmC_name), wine_dbgstr_w(buf));
2701 
2702         IStream_Release(strm_tmp);
2703     }
2704 
2705 cleanup:
2706     if(file1)
2707         IStorage_Release(file1);
2708     if(file2)
2709         IStorage_Release(file2);
2710 
2711     DeleteFileA(file1_nameA);
2712     DeleteFileA(file2_nameA);
2713 }
2714 
test_copyto_snbexclusions(void)2715 static void test_copyto_snbexclusions(void)
2716 {
2717     static const WCHAR *snb_exclude[] = {stgA_name, strmB_name, strmC_name, 0};
2718 
2719     IStorage *file1 = NULL, *file2 = NULL, *stg_tmp;
2720     IStream *strm_tmp;
2721     WCHAR buf[64];
2722     HRESULT hr;
2723 
2724     /* create & populate file1 */
2725     hr = StgCreateDocfile(file1_name, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &file1);
2726     ok(hr == S_OK, "StgCreateDocfile failed: 0x%08x\n", hr);
2727     if(FAILED(hr))
2728         goto cleanup;
2729 
2730     hr = create_test_file(file1);
2731     if(FAILED(hr))
2732         goto cleanup;
2733 
2734     /* create file2 */
2735     hr = StgCreateDocfile(file2_name, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &file2);
2736     ok(hr == S_OK, "StgCreateDocfile failed: 0x%08x\n", hr);
2737     if(FAILED(hr))
2738         goto cleanup;
2739 
2740     /* copy file1 to file2 with name exclusions */
2741     hr = IStorage_CopyTo(file1, 0, NULL, (SNB)snb_exclude, file2);
2742     ok(hr == S_OK, "CopyTo failed: 0x%08x\n", hr);
2743     if(FAILED(hr))
2744         goto cleanup;
2745 
2746     /* verify that file1 copied over, respecting exclusions */
2747     hr = IStorage_OpenStorage(file2, stgA_name, NULL,
2748             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg_tmp);
2749     ok(hr == STG_E_FILENOTFOUND, "OpenStorage should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2750     if(SUCCEEDED(hr))
2751         IStorage_Release(stg_tmp);
2752 
2753     hr = IStorage_OpenStream(file2, strmA_name, NULL,
2754             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2755     ok(hr == STG_E_FILENOTFOUND, "OpenStream should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2756     if(SUCCEEDED(hr))
2757         IStream_Release(strm_tmp);
2758 
2759     hr = IStorage_OpenStorage(file2, stgB_name, NULL,
2760             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg_tmp);
2761     ok(hr == S_OK, "OpenStorage failed: 0x%08x\n", hr);
2762 
2763     if(SUCCEEDED(hr)){
2764         hr = IStorage_OpenStream(stg_tmp, strmB_name, NULL,
2765                 STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2766         ok(hr == S_OK, "OpenStream failed: 0x%08x\n", hr);
2767 
2768         if(SUCCEEDED(hr)){
2769             memset(buf, 0, sizeof(buf));
2770             hr = IStream_Read(strm_tmp, buf, sizeof(buf), NULL);
2771             ok(hr == S_OK, "Read failed: 0x%08x\n", hr);
2772             if(SUCCEEDED(hr))
2773                 ok(strcmp_ww(buf, strmB_name) == 0,
2774                         "Expected %s to be read, got %s\n", wine_dbgstr_w(strmB_name), wine_dbgstr_w(buf));
2775 
2776             IStream_Release(strm_tmp);
2777         }
2778 
2779         IStorage_Release(stg_tmp);
2780     }
2781 
2782     hr = IStorage_OpenStream(file2, strmC_name, NULL,
2783             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2784     ok(hr == STG_E_FILENOTFOUND, "OpenStream should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2785     if(SUCCEEDED(hr))
2786         IStream_Release(strm_tmp);
2787 
2788 cleanup:
2789     if(file1)
2790         IStorage_Release(file1);
2791     if(file2)
2792         IStorage_Release(file2);
2793 
2794     DeleteFileA(file1_nameA);
2795     DeleteFileA(file2_nameA);
2796 }
2797 
test_copyto_iidexclusions_storage(void)2798 static void test_copyto_iidexclusions_storage(void)
2799 {
2800     IStorage *file1 = NULL, *file2 = NULL, *stg_tmp;
2801     IStream *strm_tmp;
2802     WCHAR buf[64];
2803     HRESULT hr;
2804 
2805     /* create & populate file1 */
2806     hr = StgCreateDocfile(file1_name, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &file1);
2807     ok(hr == S_OK, "StgCreateDocfile failed: 0x%08x\n", hr);
2808     if(FAILED(hr))
2809         goto cleanup;
2810 
2811     hr = create_test_file(file1);
2812     if(FAILED(hr))
2813         goto cleanup;
2814 
2815     /* create file2 */
2816     hr = StgCreateDocfile(file2_name, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &file2);
2817     ok(hr == S_OK, "StgCreateDocfile failed: 0x%08x\n", hr);
2818     if(FAILED(hr))
2819         goto cleanup;
2820 
2821     /* copy file1 to file2 with iid exclusions */
2822     hr = IStorage_CopyTo(file1, 1, &IID_IStorage, NULL, file2);
2823     ok(hr == S_OK, "CopyTo failed: 0x%08x\n", hr);
2824     if(FAILED(hr))
2825         goto cleanup;
2826 
2827     /* verify that file1 copied over, respecting exclusions */
2828     hr = IStorage_OpenStorage(file2, stgA_name, NULL,
2829             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg_tmp);
2830     ok(hr == STG_E_FILENOTFOUND, "OpenStorage should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2831     if(SUCCEEDED(hr))
2832         IStorage_Release(stg_tmp);
2833 
2834     hr = IStorage_OpenStream(file2, strmA_name, NULL,
2835             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2836     ok(hr == STG_E_FILENOTFOUND, "OpenStream should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2837     if(SUCCEEDED(hr))
2838         IStream_Release(strm_tmp);
2839 
2840     hr = IStorage_OpenStorage(file2, stgB_name, NULL,
2841             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg_tmp);
2842     ok(hr == STG_E_FILENOTFOUND, "OpenStorage should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2843     if(SUCCEEDED(hr))
2844         IStorage_Release(stg_tmp);
2845 
2846     hr = IStorage_OpenStream(file2, strmB_name, NULL,
2847             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2848     ok(hr == STG_E_FILENOTFOUND, "OpenStream should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2849     if(SUCCEEDED(hr))
2850         IStream_Release(strm_tmp);
2851 
2852     hr = IStorage_OpenStream(file2, strmC_name, NULL,
2853             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2854     ok(hr == S_OK, "OpenStream failed: 0x%08x\n", hr);
2855 
2856     if(SUCCEEDED(hr)){
2857         memset(buf, 0, sizeof(buf));
2858         hr = IStream_Read(strm_tmp, buf, sizeof(buf), NULL);
2859         ok(hr == S_OK, "Read failed: 0x%08x\n", hr);
2860         if(SUCCEEDED(hr))
2861             ok(strcmp_ww(buf, strmC_name) == 0,
2862                     "Expected %s to be read, got %s\n", wine_dbgstr_w(strmC_name), wine_dbgstr_w(buf));
2863 
2864         IStream_Release(strm_tmp);
2865     }
2866 
2867 cleanup:
2868     if(file1)
2869         IStorage_Release(file1);
2870     if(file2)
2871         IStorage_Release(file2);
2872 
2873     DeleteFileA(file1_nameA);
2874     DeleteFileA(file2_nameA);
2875 }
2876 
test_copyto_iidexclusions_stream(void)2877 static void test_copyto_iidexclusions_stream(void)
2878 {
2879     IStorage *file1 = NULL, *file2 = NULL, *stg_tmp;
2880     IStream *strm_tmp;
2881     HRESULT hr;
2882 
2883     /* create & populate file1 */
2884     hr = StgCreateDocfile(file1_name, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &file1);
2885     ok(hr == S_OK, "StgCreateDocfile failed: 0x%08x\n", hr);
2886     if(FAILED(hr))
2887         goto cleanup;
2888 
2889     hr = create_test_file(file1);
2890     if(FAILED(hr))
2891         goto cleanup;
2892 
2893     /* create file2 */
2894     hr = StgCreateDocfile(file2_name, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &file2);
2895     ok(hr == S_OK, "StgCreateDocfile failed: 0x%08x\n", hr);
2896     if(FAILED(hr))
2897         goto cleanup;
2898 
2899     /* copy file1 to file2 with iid exclusions */
2900     hr = IStorage_CopyTo(file1, 1, &IID_IStream, NULL, file2);
2901     ok(hr == S_OK, "CopyTo failed: 0x%08x\n", hr);
2902     if(FAILED(hr))
2903         goto cleanup;
2904 
2905     /* verify that file1 copied over, respecting exclusions */
2906     hr = IStorage_OpenStorage(file2, stgA_name, NULL,
2907             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg_tmp);
2908     ok(hr == S_OK, "OpenStorage failed: 0x%08x\n", hr);
2909 
2910     if(SUCCEEDED(hr)){
2911         hr = IStorage_OpenStream(stg_tmp, strmA_name, NULL,
2912                 STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2913         ok(hr == STG_E_FILENOTFOUND, "OpenStream should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2914         if(SUCCEEDED(hr))
2915             IStream_Release(strm_tmp);
2916 
2917         IStorage_Release(stg_tmp);
2918     }
2919 
2920     hr = IStorage_OpenStorage(file2, stgB_name, NULL,
2921             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg_tmp);
2922     ok(hr == S_OK, "OpenStorage failed: 0x%08x\n", hr);
2923 
2924     if(SUCCEEDED(hr)){
2925         hr = IStorage_OpenStream(stg_tmp, strmB_name, NULL,
2926                 STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2927         ok(hr == STG_E_FILENOTFOUND, "OpenStream should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2928         if(SUCCEEDED(hr))
2929             IStream_Release(strm_tmp);
2930 
2931         IStorage_Release(stg_tmp);
2932     }
2933 
2934     hr = IStorage_OpenStream(file2, strmC_name, NULL,
2935             STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &strm_tmp);
2936     ok(hr == STG_E_FILENOTFOUND, "OpenStream should give STG_E_FILENOTFOUND, gave: 0x%08x\n", hr);
2937     if(SUCCEEDED(hr))
2938         IStream_Release(strm_tmp);
2939 
2940 cleanup:
2941     if(file1)
2942         IStorage_Release(file1);
2943     if(file2)
2944         IStorage_Release(file2);
2945 
2946     DeleteFileA(file1_nameA);
2947     DeleteFileA(file2_nameA);
2948 }
2949 
test_rename(void)2950 static void test_rename(void)
2951 {
2952     IStorage *stg, *stg2;
2953     IStream *stm;
2954     HRESULT r;
2955     static const WCHAR stgname[] = { 'P','E','R','M','S','T','G',0 };
2956     static const WCHAR stgname2[] = { 'S','T','G',0 };
2957     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
2958     static const WCHAR stmname2[] = { 'E','N','T','S',0 };
2959     BOOL ret;
2960 
2961     DeleteFileA(filenameA);
2962 
2963     /* create the file */
2964     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
2965                             STGM_READWRITE, 0, &stg);
2966     ok(r==S_OK, "StgCreateDocfile failed\n");
2967 
2968     /* create a substorage */
2969     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
2970     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
2971 
2972     /* create a stream in the substorage */
2973     r = IStorage_CreateStream(stg2, stmname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stm);
2974     ok(r==S_OK, "IStorage->CreateStream failed, hr=%08x\n", r);
2975     IStream_Release(stm);
2976 
2977     /* rename the stream */
2978     r = IStorage_RenameElement(stg2, stmname, stmname2);
2979     ok(r==S_OK, "IStorage->RenameElement failed, hr=%08x\n", r);
2980 
2981     /* cannot open stream with old name */
2982     r = IStorage_OpenStream(stg2, stmname, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &stm);
2983     ok(r==STG_E_FILENOTFOUND, "IStorage_OpenStream should fail, hr=%08x\n", r);
2984     if (SUCCEEDED(r)) IStream_Release(stm);
2985 
2986     /* can open stream with new name */
2987     r = IStorage_OpenStream(stg2, stmname2, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &stm);
2988     ok(r==S_OK, "IStorage_OpenStream failed, hr=%08x\n", r);
2989     if (SUCCEEDED(r)) IStream_Release(stm);
2990 
2991     IStorage_Release(stg2);
2992 
2993     /* rename the storage */
2994     IStorage_RenameElement(stg, stgname, stgname2);
2995 
2996     /* cannot open storage with old name */
2997     r = IStorage_OpenStorage(stg, stgname, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg2);
2998     ok(r==STG_E_FILENOTFOUND, "IStorage_OpenStream should fail, hr=%08x\n", r);
2999     if (SUCCEEDED(r)) IStorage_Release(stg2);
3000 
3001     /* can open storage with new name */
3002     r = IStorage_OpenStorage(stg, stgname2, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg2);
3003     ok(r==S_OK, "IStorage_OpenStream should fail, hr=%08x\n", r);
3004     if (SUCCEEDED(r))
3005     {
3006         /* opened storage still has the stream */
3007         r = IStorage_OpenStream(stg2, stmname2, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &stm);
3008         ok(r==S_OK, "IStorage_OpenStream failed, hr=%08x\n", r);
3009         if (SUCCEEDED(r)) IStream_Release(stm);
3010 
3011         IStorage_Release(stg2);
3012     }
3013 
3014     IStorage_Release(stg);
3015 
3016     ret = DeleteFileA(filenameA);
3017     ok(ret, "deleted file\n");
3018 }
3019 
test_toplevel_stat(void)3020 static void test_toplevel_stat(void)
3021 {
3022     IStorage *stg = NULL;
3023     HRESULT r;
3024     STATSTG stat;
3025     char prev_dir[MAX_PATH];
3026     char temp[MAX_PATH];
3027     char full_path[MAX_PATH];
3028     LPSTR rel_pathA;
3029     WCHAR rel_path[MAX_PATH];
3030 
3031     DeleteFileA(filenameA);
3032 
3033     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
3034                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
3035     ok(r==S_OK, "StgCreateDocfile failed\n");
3036 
3037     r = IStorage_Stat( stg, &stat, STATFLAG_DEFAULT );
3038     ok(r==S_OK, "Storage_Stat failed with error 0x%08x\n", r);
3039     ok(!strcmp_ww(stat.pwcsName, filename), "expected %s, got %s\n",
3040         wine_dbgstr_w(filename), wine_dbgstr_w(stat.pwcsName));
3041     CoTaskMemFree(stat.pwcsName);
3042 
3043     IStorage_Release( stg );
3044 
3045     r = StgOpenStorage( filename, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg);
3046     ok(r==S_OK, "StgOpenStorage failed with error 0x%08x\n", r);
3047 
3048     r = IStorage_Stat( stg, &stat, STATFLAG_DEFAULT );
3049     ok(r==S_OK, "Storage_Stat failed with error 0x%08x\n", r);
3050     ok(!strcmp_ww(stat.pwcsName, filename), "expected %s, got %s\n",
3051         wine_dbgstr_w(filename), wine_dbgstr_w(stat.pwcsName));
3052     CoTaskMemFree(stat.pwcsName);
3053 
3054     IStorage_Release( stg );
3055 
3056     DeleteFileA(filenameA);
3057 
3058     /* Stat always returns the full path, even for files opened with a relative path. */
3059     GetCurrentDirectoryA(MAX_PATH, prev_dir);
3060 
3061     GetTempPathA(MAX_PATH, temp);
3062 
3063     SetCurrentDirectoryA(temp);
3064 
3065     GetFullPathNameA(filenameA, MAX_PATH, full_path, &rel_pathA);
3066     MultiByteToWideChar(CP_ACP, 0, rel_pathA, -1, rel_path, MAX_PATH);
3067 
3068     r = StgCreateDocfile( rel_path, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
3069                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
3070     ok(r==S_OK, "StgCreateDocfile failed\n");
3071 
3072     r = IStorage_Stat( stg, &stat, STATFLAG_DEFAULT );
3073     ok(r==S_OK, "Storage_Stat failed with error 0x%08x\n", r);
3074     ok(!strcmp_ww(stat.pwcsName, filename), "expected %s, got %s\n",
3075         wine_dbgstr_w(filename), wine_dbgstr_w(stat.pwcsName));
3076     CoTaskMemFree(stat.pwcsName);
3077 
3078     IStorage_Release( stg );
3079 
3080     r = StgOpenStorage( rel_path, NULL, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, NULL, 0, &stg);
3081     ok(r==S_OK, "StgOpenStorage failed with error 0x%08x\n", r);
3082 
3083     r = IStorage_Stat( stg, &stat, STATFLAG_DEFAULT );
3084     ok(r==S_OK, "Storage_Stat failed with error 0x%08x\n", r);
3085     ok(!strcmp_ww(stat.pwcsName, filename), "expected %s, got %s\n",
3086         wine_dbgstr_w(filename), wine_dbgstr_w(stat.pwcsName));
3087     CoTaskMemFree(stat.pwcsName);
3088 
3089     IStorage_Release( stg );
3090 
3091     SetCurrentDirectoryA(prev_dir);
3092 
3093     DeleteFileA(filenameA);
3094 }
3095 
test_substorage_enum(void)3096 static void test_substorage_enum(void)
3097 {
3098     IStorage *stg, *stg2;
3099     IEnumSTATSTG *ee;
3100     HRESULT r;
3101     ULONG ref;
3102     static const WCHAR stgname[] = { 'P','E','R','M','S','T','G',0 };
3103     BOOL ret;
3104 
3105     DeleteFileA(filenameA);
3106 
3107     /* create the file */
3108     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
3109                             STGM_READWRITE, 0, &stg);
3110     ok(r==S_OK, "StgCreateDocfile failed\n");
3111 
3112     /* create a substorage */
3113     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
3114     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
3115 
3116     /* create an enumelements */
3117     r = IStorage_EnumElements(stg2, 0, NULL, 0, &ee);
3118     ok(r==S_OK, "IStorage->EnumElements failed, hr=%08x\n", r);
3119 
3120     /* release the substorage */
3121     ref = IStorage_Release(stg2);
3122     todo_wine ok(ref==0, "storage not released\n");
3123 
3124     /* reopening fails, because the substorage is really still open */
3125     r = IStorage_OpenStorage(stg, stgname, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
3126     ok(r==STG_E_ACCESSDENIED, "IStorage->OpenStorage failed, hr=%08x\n", r);
3127 
3128     /* destroying the storage invalidates the enumerator */
3129     r = IStorage_DestroyElement(stg, stgname);
3130     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
3131 
3132     r = IEnumSTATSTG_Reset(ee);
3133     ok(r==STG_E_REVERTED, "IEnumSTATSTG->Reset failed, hr=%08x\n", r);
3134 
3135     IEnumSTATSTG_Release(ee);
3136 
3137     IStorage_Release(stg);
3138 
3139     ret = DeleteFileA(filenameA);
3140     ok(ret, "deleted file\n");
3141 }
3142 
test_copyto_locking(void)3143 static void test_copyto_locking(void)
3144 {
3145     IStorage *stg, *stg2, *stg3, *stg4;
3146     IStream *stm;
3147     HRESULT r;
3148     static const WCHAR stgname[] = { 'S','T','G','1',0 };
3149     static const WCHAR stgname2[] = { 'S','T','G','2',0 };
3150     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
3151     BOOL ret;
3152 
3153     DeleteFileA(filenameA);
3154 
3155     /* create the file */
3156     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
3157                             STGM_READWRITE, 0, &stg);
3158     ok(r==S_OK, "StgCreateDocfile failed\n");
3159 
3160     /* create a substorage */
3161     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
3162     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
3163 
3164     /* create another substorage */
3165     r = IStorage_CreateStorage(stg, stgname2, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg3);
3166     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
3167 
3168     /* add a stream, and leave it open */
3169     r = IStorage_CreateStream(stg2, stmname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stm);
3170     ok(r==S_OK, "IStorage->CreateStream failed, hr=%08x\n", r);
3171 
3172     /* Try to copy the storage while the stream is open */
3173     r = IStorage_CopyTo(stg2, 0, NULL, NULL, stg3);
3174     ok(r==S_OK, "IStorage->CopyTo failed, hr=%08x\n", r);
3175 
3176     IStream_Release(stm);
3177 
3178     /* create a substorage */
3179     r = IStorage_CreateStorage(stg2, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg4);
3180     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
3181 
3182     /* Try to copy the storage while the substorage is open */
3183     r = IStorage_CopyTo(stg2, 0, NULL, NULL, stg3);
3184     ok(r==S_OK, "IStorage->CopyTo failed, hr=%08x\n", r);
3185 
3186     IStorage_Release(stg4);
3187     IStorage_Release(stg3);
3188     IStorage_Release(stg2);
3189     IStorage_Release(stg);
3190 
3191     ret = DeleteFileA(filenameA);
3192     ok(ret, "deleted file\n");
3193 }
3194 
test_copyto_recursive(void)3195 static void test_copyto_recursive(void)
3196 {
3197     IStorage *stg, *stg2, *stg3, *stg4;
3198     HRESULT r;
3199     static const WCHAR stgname[] = { 'S','T','G','1',0 };
3200     static const WCHAR stgname2[] = { 'S','T','G','2',0 };
3201     BOOL ret;
3202 
3203     DeleteFileA(filenameA);
3204 
3205     /* create the file */
3206     r = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE |
3207                             STGM_READWRITE, 0, &stg);
3208     ok(r==S_OK, "StgCreateDocfile failed\n");
3209 
3210     /* create a substorage */
3211     r = IStorage_CreateStorage(stg, stgname, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stg2);
3212     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
3213 
3214     /* copy the parent to the child */
3215     r = IStorage_CopyTo(stg, 0, NULL, NULL, stg2);
3216     ok(r==STG_E_ACCESSDENIED, "IStorage->CopyTo failed, hr=%08x\n", r);
3217 
3218     /* create a transacted substorage */
3219     r = IStorage_CreateStorage(stg, stgname2, STGM_READWRITE | STGM_SHARE_EXCLUSIVE | STGM_TRANSACTED, 0, 0, &stg3);
3220     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
3221 
3222     /* copy the parent to the transacted child */
3223     r = IStorage_CopyTo(stg, 0, NULL, NULL, stg2);
3224     ok(r==STG_E_ACCESSDENIED, "IStorage->CopyTo failed, hr=%08x\n", r);
3225 
3226     /* create a transacted subsubstorage */
3227     r = IStorage_CreateStorage(stg3, stgname2, STGM_READWRITE | STGM_SHARE_EXCLUSIVE | STGM_TRANSACTED, 0, 0, &stg4);
3228     ok(r==S_OK, "IStorage->CreateStorage failed, hr=%08x\n", r);
3229 
3230     /* copy the parent to the transacted child of the transacted child */
3231     r = IStorage_CopyTo(stg, 0, NULL, NULL, stg4);
3232     ok(r==STG_E_ACCESSDENIED, "IStorage->CopyTo failed, hr=%08x\n", r);
3233 
3234     /* copy the parent but exclude storage objects */
3235     r = IStorage_CopyTo(stg, 1, &IID_IStorage, NULL, stg4);
3236     ok(r==S_OK, "IStorage->CopyTo failed, hr=%08x\n", r);
3237 
3238     IStorage_Release(stg4);
3239     IStorage_Release(stg3);
3240     IStorage_Release(stg2);
3241     IStorage_Release(stg);
3242 
3243     ret = DeleteFileA(filenameA);
3244     ok(ret, "deleted file\n");
3245 }
3246 
test_hglobal_storage_creation(void)3247 static void test_hglobal_storage_creation(void)
3248 {
3249     ILockBytes *ilb = NULL;
3250     IStorage *stg = NULL;
3251     HRESULT r;
3252     STATSTG stat;
3253     char junk[512];
3254     ULARGE_INTEGER offset;
3255 
3256     r = CreateILockBytesOnHGlobal(NULL, TRUE, &ilb);
3257     ok(r == S_OK, "CreateILockBytesOnHGlobal failed, hr=%x\n", r);
3258 
3259     offset.QuadPart = 0;
3260     memset(junk, 0xaa, 512);
3261     r = ILockBytes_WriteAt(ilb, offset, junk, 512, NULL);
3262     ok(r == S_OK, "ILockBytes_WriteAt failed, hr=%x\n", r);
3263 
3264     offset.QuadPart = 2000;
3265     r = ILockBytes_WriteAt(ilb, offset, junk, 512, NULL);
3266     ok(r == S_OK, "ILockBytes_WriteAt failed, hr=%x\n", r);
3267 
3268     r = StgCreateDocfileOnILockBytes(ilb, STGM_CREATE|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0,  &stg);
3269     ok(r == S_OK, "StgCreateDocfileOnILockBytes failed, hr=%x\n", r);
3270 
3271     IStorage_Release(stg);
3272 
3273     r = StgOpenStorageOnILockBytes(ilb, NULL, STGM_READ|STGM_SHARE_EXCLUSIVE,
3274         NULL, 0, &stg);
3275     ok(r == S_OK, "StgOpenStorageOnILockBytes failed, hr=%x\n", r);
3276 
3277     if (SUCCEEDED(r))
3278     {
3279         r = IStorage_Stat(stg, &stat, STATFLAG_NONAME);
3280         ok(r == S_OK, "StgOpenStorageOnILockBytes failed, hr=%x\n", r);
3281         ok(IsEqualCLSID(&stat.clsid, &GUID_NULL), "unexpected CLSID value\n");
3282 
3283         IStorage_Release(stg);
3284     }
3285 
3286     r = ILockBytes_Stat(ilb, &stat, STATFLAG_NONAME);
3287     ok(r == S_OK, "ILockBytes_Stat failed, hr=%x\n", r);
3288     ok(stat.cbSize.u.LowPart < 2512, "expected truncated size, got %d\n", stat.cbSize.u.LowPart);
3289 
3290     ILockBytes_Release(ilb);
3291 }
3292 
test_convert(void)3293 static void test_convert(void)
3294 {
3295     static const WCHAR filename[] = {'s','t','o','r','a','g','e','.','s','t','g',0};
3296     IStorage *stg;
3297     HRESULT hr;
3298 
3299     hr = GetConvertStg(NULL);
3300     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3301 
3302     hr = StgCreateDocfile( filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stg);
3303     ok(hr == S_OK, "StgCreateDocfile failed\n");
3304     hr = GetConvertStg(stg);
3305     ok(hr == STG_E_FILENOTFOUND, "got 0x%08x\n", hr);
3306     hr = SetConvertStg(stg, TRUE);
3307     ok(hr == S_OK, "got 0x%08x\n", hr);
3308     hr = SetConvertStg(stg, TRUE);
3309     ok(hr == S_OK, "got 0x%08x\n", hr);
3310     hr = GetConvertStg(stg);
3311     ok(hr == S_OK, "got 0x%08x\n", hr);
3312     hr = SetConvertStg(stg, FALSE);
3313     ok(hr == S_OK, "got 0x%08x\n", hr);
3314     hr = GetConvertStg(stg);
3315     ok(hr == S_FALSE, "got 0x%08x\n", hr);
3316 
3317     IStorage_Release(stg);
3318 
3319     DeleteFileW(filename);
3320 }
3321 
test_direct_swmr(void)3322 static void test_direct_swmr(void)
3323 {
3324     static const WCHAR fileW[] = {'w','i','n','e','t','e','s','t',0};
3325     IDirectWriterLock *dwlock;
3326     ULONG ref, ref2;
3327     IStorage *stg;
3328     HRESULT hr;
3329 
3330     /* it's possible to create in writer mode */
3331     hr = StgCreateDocfile(fileW, STGM_CREATE | STGM_READWRITE | STGM_SHARE_DENY_WRITE | STGM_DIRECT_SWMR, 0, &stg);
3332 todo_wine
3333     ok(hr == S_OK, "got %08x\n", hr);
3334 if (hr == S_OK) {
3335     IStorage_Release(stg);
3336     DeleteFileW(fileW);
3337 }
3338 
3339     hr = StgCreateDocfile(fileW, STGM_CREATE | STGM_READWRITE | STGM_SHARE_DENY_WRITE | STGM_TRANSACTED, 0, &stg);
3340     ok(hr == S_OK, "got %08x\n", hr);
3341     IStorage_Release(stg);
3342 
3343     /* reader mode */
3344     hr = StgOpenStorage(fileW, NULL, STGM_DIRECT_SWMR | STGM_READ | STGM_SHARE_DENY_NONE, NULL, 0, &stg);
3345     ok(hr == S_OK || broken(hr == STG_E_INVALIDFLAG), "got %08x\n", hr);
3346     if(hr == S_OK)
3347     {
3348        hr = IStorage_QueryInterface(stg, &IID_IDirectWriterLock, (void**)&dwlock);
3349        ok(hr == E_NOINTERFACE, "got %08x\n", hr);
3350        IStorage_Release(stg);
3351     }
3352 
3353     /* writer mode */
3354     hr = StgOpenStorage(fileW, NULL, STGM_DIRECT_SWMR | STGM_READWRITE | STGM_SHARE_DENY_WRITE, NULL, 0, &stg);
3355     ok(hr == S_OK, "got %08x\n", hr);
3356     if(hr == S_OK)
3357     {
3358         ref = IStorage_AddRef(stg);
3359         IStorage_Release(stg);
3360 
3361         hr = IStorage_QueryInterface(stg, &IID_IDirectWriterLock, (void**)&dwlock);
3362         ok(hr == S_OK, "got %08x\n", hr);
3363 
3364         ref2 = IStorage_AddRef(stg);
3365         IStorage_Release(stg);
3366         ok(ref2 == ref + 1, "got %u\n", ref2);
3367 
3368         IDirectWriterLock_Release(dwlock);
3369         IStorage_Release(stg);
3370     }
3371 
3372     DeleteFileW(fileW);
3373 }
3374 
3375 struct lock_test
3376 {
3377     DWORD stg_mode;
3378     BOOL create;
3379     DWORD access;
3380     DWORD sharing;
3381     const int *locked_bytes;
3382     const int *fail_ranges;
3383     BOOL todo;
3384 };
3385 
3386 static const int priority_locked_bytes[] = { 0x158, 0x181, 0x193, -1 };
3387 static const int rwex_locked_bytes[] = { 0x193, 0x1a7, 0x1bb, 0x1cf, -1 };
3388 static const int rw_locked_bytes[] = { 0x193, 0x1a7, -1 };
3389 static const int nosn_locked_bytes[] = { 0x16c, 0x193, 0x1a7, 0x1cf, -1 };
3390 static const int rwdw_locked_bytes[] = { 0x193, 0x1a7, 0x1cf, -1 };
3391 static const int wodw_locked_bytes[] = { 0x1a7, 0x1cf, -1 };
3392 static const int tr_locked_bytes[] = { 0x193, -1 };
3393 static const int no_locked_bytes[] = { -1 };
3394 static const int roex_locked_bytes[] = { 0x193, 0x1bb, 0x1cf, -1 };
3395 
3396 static const int rwex_fail_ranges[] = { 0x193,0x1e3, -1 };
3397 static const int rw_fail_ranges[] = { 0x1bb,0x1e3, -1 };
3398 static const int rwdw_fail_ranges[] = { 0x1a7,0x1e3, -1 };
3399 static const int dw_fail_ranges[] = { 0x1a7,0x1cf, -1 };
3400 static const int tr_fail_ranges[] = { 0x1bb,0x1cf, -1 };
3401 static const int pr_fail_ranges[] = { 0x180,0x181, 0x1bb,0x1cf, -1 };
3402 static const int roex_fail_ranges[] = { 0x0,-1 };
3403 
3404 static const struct lock_test lock_tests[] = {
3405     { STGM_PRIORITY, FALSE, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE, priority_locked_bytes, pr_fail_ranges, FALSE },
3406     { STGM_CREATE|STGM_SHARE_EXCLUSIVE|STGM_READWRITE, TRUE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ, rwex_locked_bytes, 0, FALSE },
3407     { STGM_CREATE|STGM_SHARE_EXCLUSIVE|STGM_READWRITE|STGM_TRANSACTED, TRUE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ, rwex_locked_bytes, 0, FALSE },
3408     { STGM_CREATE|STGM_READWRITE|STGM_TRANSACTED, TRUE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ|FILE_SHARE_WRITE, rw_locked_bytes, 0, FALSE },
3409     { STGM_CREATE|STGM_READWRITE|STGM_SHARE_DENY_WRITE|STGM_TRANSACTED, TRUE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ, rwdw_locked_bytes, 0, FALSE },
3410     { STGM_CREATE|STGM_WRITE|STGM_SHARE_DENY_WRITE|STGM_TRANSACTED, TRUE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ, wodw_locked_bytes, 0, FALSE },
3411     { STGM_SHARE_EXCLUSIVE|STGM_READWRITE, FALSE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ, rwex_locked_bytes, rwex_fail_ranges, FALSE },
3412     { STGM_SHARE_EXCLUSIVE|STGM_READWRITE|STGM_TRANSACTED, FALSE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ, rwex_locked_bytes, rwex_fail_ranges, FALSE },
3413     { STGM_READWRITE|STGM_TRANSACTED, FALSE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ|FILE_SHARE_WRITE, rw_locked_bytes, rw_fail_ranges, FALSE },
3414     { STGM_READWRITE|STGM_TRANSACTED|STGM_NOSNAPSHOT, FALSE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ|FILE_SHARE_WRITE, nosn_locked_bytes, rwdw_fail_ranges, FALSE },
3415     { STGM_READWRITE|STGM_TRANSACTED|STGM_SHARE_DENY_WRITE, FALSE, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ, rwdw_locked_bytes, rwdw_fail_ranges, FALSE },
3416     { STGM_READ|STGM_SHARE_DENY_WRITE, FALSE, GENERIC_READ, FILE_SHARE_READ, no_locked_bytes, dw_fail_ranges, TRUE },
3417     { STGM_READ|STGM_TRANSACTED, FALSE, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE, tr_locked_bytes, tr_fail_ranges, FALSE },
3418     { STGM_READ|STGM_SHARE_EXCLUSIVE, FALSE, GENERIC_READ, FILE_SHARE_READ, roex_locked_bytes, roex_fail_ranges, FALSE },
3419     { STGM_READ|STGM_SHARE_EXCLUSIVE|STGM_TRANSACTED, FALSE, GENERIC_READ, FILE_SHARE_READ, roex_locked_bytes, roex_fail_ranges, FALSE },
3420 };
3421 
can_open(LPCWSTR filename,DWORD access,DWORD sharing)3422 static BOOL can_open(LPCWSTR filename, DWORD access, DWORD sharing)
3423 {
3424     HANDLE hfile;
3425 
3426     hfile = CreateFileW(filename, access, sharing, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
3427 
3428     if (hfile == INVALID_HANDLE_VALUE)
3429         return FALSE;
3430 
3431     CloseHandle(hfile);
3432     return TRUE;
3433 }
3434 
check_sharing(LPCWSTR filename,const struct lock_test * current,DWORD access,DWORD sharing,const char * desc,DWORD * open_mode,BOOL * any_failure)3435 static void check_sharing(LPCWSTR filename, const struct lock_test *current,
3436     DWORD access, DWORD sharing, const char *desc, DWORD *open_mode, BOOL *any_failure)
3437 {
3438     if (can_open(filename, access, FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE))
3439     {
3440         *open_mode = access;
3441         if (!current->todo || (current->sharing & sharing))
3442             ok(current->sharing & sharing ||
3443                 broken(!(current->sharing & sharing) && access == GENERIC_WRITE && (current->stg_mode & 0xf) != STGM_READ) /* win2k */,
3444                 "file with mode %x should not be openable with %s permission\n", current->stg_mode, desc);
3445         else
3446         {
3447             todo_wine ok(current->sharing & sharing ||
3448                 broken(!(current->sharing & sharing) && access == GENERIC_WRITE && (current->stg_mode & 0xf) != STGM_READ) /* win2k */,
3449                 "file with mode %x should not be openable with %s permission\n", current->stg_mode, desc);
3450             *any_failure = TRUE;
3451         }
3452     }
3453     else
3454     {
3455         if (!current->todo || !(current->sharing & sharing))
3456             ok(!(current->sharing & sharing), "file with mode %x should be openable with %s permission\n", current->stg_mode, desc);
3457         else
3458         {
3459             todo_wine ok(!(current->sharing & sharing), "file with mode %x should be openable with %s permission\n", current->stg_mode, desc);
3460             *any_failure = TRUE;
3461         }
3462     }
3463 }
3464 
check_access(LPCWSTR filename,const struct lock_test * current,DWORD access,DWORD sharing,const char * desc,DWORD open_mode,BOOL * any_failure)3465 static void check_access(LPCWSTR filename, const struct lock_test *current,
3466     DWORD access, DWORD sharing, const char *desc, DWORD open_mode, BOOL *any_failure)
3467 {
3468     if (can_open(filename, open_mode, (FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE) & ~sharing))
3469     {
3470         if (!current->todo || !(current->access & access))
3471             ok(!(current->access & access), "file with mode %x should not be openable without %s sharing\n", current->stg_mode, desc);
3472         else
3473         {
3474             todo_wine ok(!(current->access & access), "file with mode %x should not be openable without %s sharing\n", current->stg_mode, desc);
3475             *any_failure = TRUE;
3476         }
3477     }
3478     else
3479     {
3480         if (!current->todo || (current->access & access))
3481             ok(current->access & access, "file with mode %x should be openable without %s sharing\n", current->stg_mode, desc);
3482         else
3483         {
3484             todo_wine ok(current->access & access, "file with mode %x should be openable without %s sharing\n", current->stg_mode, desc);
3485             *any_failure = TRUE;
3486         }
3487     }
3488 }
3489 
test_locking(void)3490 static void test_locking(void)
3491 {
3492     static const WCHAR filename[] = {'w','i','n','e','t','e','s','t',0};
3493     int i;
3494     IStorage *stg;
3495     HRESULT hr;
3496 
3497     for (i = 0; i < ARRAY_SIZE(lock_tests); i++)
3498     {
3499         const struct lock_test *current = &lock_tests[i];
3500         BOOL any_failure = FALSE;
3501         DWORD open_mode = 0;
3502 
3503         if (current->create)
3504         {
3505             hr = StgCreateDocfile(filename, current->stg_mode, 0, &stg);
3506             ok(SUCCEEDED(hr), "StgCreateDocfile with mode %x failed with hr %x\n", current->stg_mode, hr);
3507             if (FAILED(hr)) continue;
3508         }
3509         else
3510         {
3511             hr = StgCreateDocfile(filename, STGM_CREATE | STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stg);
3512             ok(SUCCEEDED(hr), "StgCreateDocfile failed with hr %x\n", hr);
3513             if (FAILED(hr)) continue;
3514             IStorage_Release(stg);
3515 
3516             hr = StgOpenStorage(filename, NULL, current->stg_mode, NULL, 0, &stg);
3517             ok(SUCCEEDED(hr), "StgOpenStorage with mode %x failed with hr %x\n", current->stg_mode, hr);
3518             if (FAILED(hr))
3519             {
3520                 DeleteFileW(filename);
3521                 continue;
3522             }
3523         }
3524 
3525         check_sharing(filename, current, GENERIC_READ, FILE_SHARE_READ, "READ", &open_mode, &any_failure);
3526         check_sharing(filename, current, GENERIC_WRITE, FILE_SHARE_WRITE, "WRITE", &open_mode, &any_failure);
3527         check_sharing(filename, current, DELETE, FILE_SHARE_DELETE, "DELETE", &open_mode, &any_failure);
3528 
3529         if (open_mode != 0)
3530         {
3531             HANDLE hfile;
3532             BOOL locked, expect_locked;
3533             OVERLAPPED ol;
3534             const int* next_lock = current->locked_bytes;
3535 
3536             check_access(filename, current, GENERIC_READ, FILE_SHARE_READ, "READ", open_mode, &any_failure);
3537             check_access(filename, current, GENERIC_WRITE, FILE_SHARE_WRITE, "WRITE", open_mode, &any_failure);
3538             check_access(filename, current, DELETE, FILE_SHARE_DELETE, "DELETE", open_mode, &any_failure);
3539 
3540             hfile = CreateFileW(filename, open_mode, FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
3541             ok(hfile != INVALID_HANDLE_VALUE, "couldn't open file with mode %x\n", current->stg_mode);
3542 
3543             ol.OffsetHigh = 0;
3544             ol.hEvent = NULL;
3545 
3546             for (ol.Offset = 0x7ffffe00; ol.Offset != 0x80000000; ol.Offset++)
3547             {
3548                 if (LockFileEx(hfile, LOCKFILE_EXCLUSIVE_LOCK|LOCKFILE_FAIL_IMMEDIATELY, 0, 1, 0, &ol))
3549                     locked = FALSE;
3550                 else
3551                 {
3552                     ok(!LockFileEx(hfile, LOCKFILE_FAIL_IMMEDIATELY, 0, 1, 0, &ol), "shared locks should not be used\n");
3553                     locked = TRUE;
3554                 }
3555 
3556                 UnlockFileEx(hfile, 0, 1, 0, &ol);
3557 
3558                 if ((ol.Offset&0x1ff) == *next_lock)
3559                 {
3560                     expect_locked = TRUE;
3561                     next_lock++;
3562                 }
3563                 else
3564                     expect_locked = FALSE;
3565 
3566                 if (!current->todo || locked == expect_locked)
3567                     ok(locked == expect_locked, "byte %x of file with mode %x is %slocked but should %sbe\n",
3568                        ol.Offset, current->stg_mode, locked?"":"not ", expect_locked?"":"not ");
3569                 else
3570                 {
3571                     any_failure = TRUE;
3572                     todo_wine ok(locked == expect_locked, "byte %x of file with mode %x is %slocked but should %sbe\n",
3573                               ol.Offset, current->stg_mode, locked?"":"not ", expect_locked?"":"not ");
3574                 }
3575             }
3576 
3577             CloseHandle(hfile);
3578         }
3579 
3580         IStorage_Release( stg );
3581 
3582         if (!current->create)
3583         {
3584             HANDLE hfile;
3585             BOOL failed, expect_failed=FALSE;
3586             OVERLAPPED ol;
3587             const int* next_range = current->fail_ranges;
3588 
3589             hfile = CreateFileW(filename, open_mode, FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
3590             ok(hfile != INVALID_HANDLE_VALUE, "couldn't open file with mode %x\n", current->stg_mode);
3591 
3592             ol.OffsetHigh = 0;
3593             ol.hEvent = NULL;
3594 
3595             for (ol.Offset = 0x7ffffe00; ol.Offset != 0x80000000; ol.Offset++)
3596             {
3597                 if (ol.Offset == 0x7fffff92 ||
3598                     (ol.Offset == 0x7fffff80 && current->stg_mode == (STGM_TRANSACTED|STGM_READWRITE)) ||
3599                     (ol.Offset == 0x7fffff80 && current->stg_mode == (STGM_TRANSACTED|STGM_READ)))
3600                     continue; /* This makes opens hang */
3601 
3602                 if (ol.Offset < 0x7fffff00)
3603                     LockFileEx(hfile, 0, 0, 1, 0, &ol);
3604                 else
3605                     LockFileEx(hfile, LOCKFILE_EXCLUSIVE_LOCK, 0, 1, 0, &ol);
3606 
3607                 hr = StgOpenStorage(filename, NULL, current->stg_mode, NULL, 0, &stg);
3608                 ok(hr == S_OK || hr == STG_E_LOCKVIOLATION || hr == STG_E_SHAREVIOLATION, "failed with unexpected hr %x\n", hr);
3609                 if (SUCCEEDED(hr)) IStorage_Release(stg);
3610 
3611                 UnlockFileEx(hfile, 0, 1, 0, &ol);
3612 
3613                 failed = FAILED(hr);
3614 
3615                 if (!expect_failed && (ol.Offset&0x1ff) == next_range[0])
3616                 {
3617                     expect_failed = TRUE;
3618                 }
3619                 else if (expect_failed && (ol.Offset&0x1ff) == next_range[1])
3620                 {
3621                     expect_failed = FALSE;
3622                     next_range += 2;
3623                 }
3624 
3625                 if (!current->todo || failed == expect_failed)
3626                     ok(failed == expect_failed, "open with byte %x locked, mode %x %s but should %s\n",
3627                        ol.Offset, current->stg_mode, failed?"failed":"succeeded", expect_failed?"fail":"succeed");
3628                 else
3629                 {
3630                     any_failure = TRUE;
3631                     todo_wine ok(failed == expect_failed, "open with byte %x locked, mode %x %s but should %s\n",
3632                                  ol.Offset, current->stg_mode, failed?"failed":"succeeded", expect_failed?"fail":"succeed");
3633                 }
3634             }
3635 
3636             CloseHandle(hfile);
3637         }
3638 
3639         DeleteFileW(filename);
3640 
3641         if (current->todo && !any_failure)
3642             todo_wine ok(1, "tests succeeded for mode %x\n", current->stg_mode);
3643     }
3644 }
3645 
test_transacted_shared(void)3646 static void test_transacted_shared(void)
3647 {
3648     IStorage *stg = NULL;
3649     IStorage *stgrw = NULL;
3650     HRESULT r;
3651     IStream *stm = NULL;
3652     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
3653     LARGE_INTEGER pos;
3654     ULARGE_INTEGER upos;
3655     char buffer[10];
3656     ULONG bytesread;
3657 
3658     DeleteFileA(filenameA);
3659 
3660     /* create a new transacted storage with a stream */
3661     r = StgCreateDocfile(filename, STGM_CREATE |
3662                             STGM_READWRITE |STGM_TRANSACTED, 0, &stg);
3663     ok(r==S_OK, "StgCreateDocfile failed %x\n", r);
3664 
3665     r = WriteClassStg(stg, &test_stg_cls);
3666     ok(r == S_OK, "WriteClassStg failed %x\n", r);
3667 
3668     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm);
3669     ok(r==S_OK, "IStorage->CreateStream failed %x\n", r);
3670 
3671     pos.QuadPart = 0;
3672     r = IStream_Seek(stm, pos, 0, &upos);
3673     ok(r==S_OK, "IStream->Seek failed %x\n", r);
3674 
3675     r = IStream_Write(stm, "aaa", 3, NULL);
3676     ok(r==S_OK, "IStream->Write failed %x\n", r);
3677 
3678     r = IStorage_Commit(stg, STGC_ONLYIFCURRENT);
3679     ok(r==S_OK, "IStorage->Commit failed %x\n", r);
3680 
3681     /* open a second transacted read/write storage */
3682     r = StgOpenStorage(filename, NULL, STGM_READWRITE | STGM_TRANSACTED | STGM_SHARE_DENY_NONE, NULL, 0, &stgrw);
3683     ok(r==S_OK, "StgOpenStorage failed %x\n", r);
3684 
3685     /* update stream on the first storage and commit */
3686     pos.QuadPart = 0;
3687     r = IStream_Seek(stm, pos, 0, &upos);
3688     ok(r==S_OK, "IStream->Seek failed %x\n", r);
3689 
3690     r = IStream_Write(stm, "ccc", 3, NULL);
3691     ok(r==S_OK, "IStream->Write failed %x\n", r);
3692 
3693     r = IStorage_Commit(stg, STGC_ONLYIFCURRENT);
3694     ok(r==S_OK, "IStorage->Commit failed %x\n", r);
3695 
3696     /* update again without committing */
3697     pos.QuadPart = 0;
3698     r = IStream_Seek(stm, pos, 0, &upos);
3699     ok(r==S_OK, "IStream->Seek failed %x\n", r);
3700 
3701     r = IStream_Write(stm, "ddd", 3, NULL);
3702     ok(r==S_OK, "IStream->Write failed %x\n", r);
3703 
3704     IStream_Release(stm);
3705 
3706     /* we can still read the old content from the second storage */
3707     r = IStorage_OpenStream(stgrw, stmname, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &stm);
3708     ok(r==S_OK, "IStorage->OpenStream failed %x\n", r);
3709 
3710     pos.QuadPart = 0;
3711     r = IStream_Seek(stm, pos, 0, &upos);
3712     ok(r==S_OK, "IStream->Seek failed %x\n", r);
3713 
3714     r = IStream_Read(stm, buffer, sizeof(buffer), &bytesread);
3715     ok(r==S_OK, "IStream->Read failed %x\n", r);
3716     ok(bytesread == 3, "read wrong number of bytes %i\n", bytesread);
3717     ok(memcmp(buffer, "aaa", 3) == 0, "wrong data\n");
3718 
3719     /* and overwrite the data */
3720     pos.QuadPart = 0;
3721     r = IStream_Seek(stm, pos, 0, &upos);
3722     ok(r==S_OK, "IStream->Seek failed %x\n", r);
3723 
3724     r = IStream_Write(stm, "bbb", 3, NULL);
3725     ok(r==S_OK, "IStream->Write failed %x\n", r);
3726 
3727     IStream_Release(stm);
3728 
3729     /* commit fails because we're out of date */
3730     r = IStorage_Commit(stgrw, STGC_ONLYIFCURRENT);
3731     ok(r==STG_E_NOTCURRENT, "IStorage->Commit failed %x\n", r);
3732 
3733     /* unless we force it */
3734     r = IStorage_Commit(stgrw, STGC_DEFAULT);
3735     ok(r==S_OK, "IStorage->Commit failed %x\n", r);
3736 
3737     /* reverting gets us back to the last commit from the same storage */
3738     r = IStorage_Revert(stg);
3739     ok(r==S_OK, "IStorage->Revert failed %x\n", r);
3740 
3741     r = IStorage_OpenStream(stg, stmname, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &stm);
3742     ok(r==S_OK, "IStorage->CreateStream failed %x\n", r);
3743 
3744     pos.QuadPart = 0;
3745     r = IStream_Seek(stm, pos, 0, &upos);
3746     ok(r==S_OK, "IStream->Seek failed %x\n", r);
3747 
3748     r = IStream_Read(stm, buffer, sizeof(buffer), &bytesread);
3749     ok(r==S_OK, "IStream->Read failed %x\n", r);
3750     ok(bytesread == 3, "read wrong number of bytes %i\n", bytesread);
3751     ok(memcmp(buffer, "ccc", 3) == 0, "wrong data\n");
3752 
3753     /* and committing fails forever */
3754     r = IStorage_Commit(stg, STGC_ONLYIFCURRENT);
3755     ok(r==STG_E_NOTCURRENT, "IStorage->Commit failed %x\n", r);
3756 
3757     IStream_Release(stm);
3758 
3759     IStorage_Release(stg);
3760     IStorage_Release(stgrw);
3761 
3762     DeleteFileA(filenameA);
3763 }
3764 
test_overwrite(void)3765 static void test_overwrite(void)
3766 {
3767     IStorage *stg = NULL;
3768     HRESULT r;
3769     IStream *stm = NULL;
3770     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
3771     static const WCHAR stmname2[] = { 'C','O','N','T','E','N','T','2',0 };
3772     LARGE_INTEGER pos;
3773     ULARGE_INTEGER upos;
3774     char buffer[4096];
3775     DWORD orig_size, new_size;
3776     ULONG bytesread;
3777     HANDLE hfile;
3778     int i;
3779 
3780     DeleteFileA(filenameA);
3781 
3782     r = StgCreateDocfile(filename, STGM_CREATE | STGM_READWRITE | STGM_TRANSACTED, 0, &stg);
3783     ok(r==S_OK, "StgCreateDocfile failed %x\n", r);
3784 
3785     r = WriteClassStg(stg, &test_stg_cls);
3786     ok(r == S_OK, "WriteClassStg failed %x\n", r);
3787 
3788     r = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, 0, &stm);
3789     ok(r==S_OK, "IStorage->CreateStream failed %x\n", r);
3790 
3791     pos.QuadPart = 0;
3792     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &upos);
3793     ok(r==S_OK, "IStream->Seek failed %x\n", r);
3794 
3795     memset(buffer, 'a', sizeof(buffer));
3796     for (i=0; i<4; i++)
3797     {
3798         /* Write enough bytes to pass the minimum storage file size */
3799         r = IStream_Write(stm, buffer, sizeof(buffer), NULL);
3800         ok(r==S_OK, "IStream->Write failed %x\n", r);
3801     }
3802 
3803     r = IStorage_Commit(stg, STGC_DEFAULT);
3804     ok(r==S_OK, "IStorage->Commit failed %x\n", r);
3805 
3806     hfile = CreateFileA(filenameA, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
3807         NULL, OPEN_EXISTING, 0, NULL);
3808     ok(hfile != NULL, "couldn't open file %d\n", GetLastError());
3809 
3810     orig_size = GetFileSize(hfile, NULL);
3811 
3812     pos.QuadPart = 0;
3813     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &upos);
3814     ok(r==S_OK, "IStream->Seek failed %x\n", r);
3815 
3816     r = IStream_Write(stm, "b", 1, NULL);
3817     ok(r==S_OK, "IStream->Write failed %x\n", r);
3818 
3819     r = IStorage_Commit(stg, STGC_OVERWRITE);
3820     ok(r==S_OK, "IStorage->Commit failed %x\n", r);
3821 
3822     new_size = GetFileSize(hfile, NULL);
3823 
3824     todo_wine ok(new_size == orig_size, "file grew from %d bytes to %d\n", orig_size, new_size);
3825 
3826     IStream_Release(stm);
3827 
3828     IStorage_RenameElement(stg, stmname, stmname2);
3829 
3830     r = IStorage_Commit(stg, STGC_OVERWRITE);
3831     ok(r==S_OK, "IStorage->Commit failed %x\n", r);
3832 
3833     new_size = GetFileSize(hfile, NULL);
3834 
3835     todo_wine ok(new_size == orig_size, "file grew from %d bytes to %d\n", orig_size, new_size);
3836 
3837     IStorage_Release(stg);
3838 
3839     r = StgOpenStorage(filename, NULL, STGM_READWRITE | STGM_TRANSACTED | STGM_SHARE_EXCLUSIVE, NULL, 0, &stg);
3840     ok(r==S_OK, "StgOpenStorage failed %x\n", r);
3841 
3842     r = IStorage_OpenStream(stg, stmname2, NULL, STGM_SHARE_EXCLUSIVE | STGM_READWRITE, 0, &stm);
3843     ok(r==S_OK, "IStorage->CreateStream failed %x\n", r);
3844 
3845     r = IStream_Read(stm, buffer, sizeof(buffer), &bytesread);
3846     ok(r==S_OK, "IStream->Write failed %x\n", r);
3847     ok(bytesread == sizeof(buffer), "only read %d bytes\n", bytesread);
3848     ok(buffer[0] == 'b', "unexpected data at byte 0\n");
3849 
3850     for (i=1; i<sizeof(buffer); i++)
3851         if (buffer[i] != 'a')
3852             break;
3853     ok(i == sizeof(buffer), "unexpected data at byte %i\n", i);
3854 
3855     pos.QuadPart = 0;
3856     r = IStream_Seek(stm, pos, STREAM_SEEK_SET, &upos);
3857     ok(r==S_OK, "IStream->Seek failed %x\n", r);
3858 
3859     r = IStream_Write(stm, "c", 1, NULL);
3860     ok(r==S_OK, "IStream->Write failed %x\n", r);
3861 
3862     r = IStorage_Commit(stg, STGC_OVERWRITE);
3863     ok(r==S_OK, "IStorage->Commit failed %x\n", r);
3864 
3865     new_size = GetFileSize(hfile, NULL);
3866 
3867     todo_wine ok(new_size == orig_size, "file grew from %d bytes to %d\n", orig_size, new_size);
3868 
3869     IStream_Release(stm);
3870 
3871     IStorage_Release(stg);
3872 
3873     CloseHandle(hfile);
3874 
3875     DeleteFileA(filenameA);
3876 }
3877 
test_custom_lockbytes(void)3878 static void test_custom_lockbytes(void)
3879 {
3880     static const WCHAR stmname[] = { 'C','O','N','T','E','N','T','S',0 };
3881     TestLockBytes* lockbytes;
3882     HRESULT hr;
3883     IStorage* stg;
3884     IStream* stm;
3885 
3886     CreateTestLockBytes(&lockbytes);
3887 
3888     hr = StgCreateDocfileOnILockBytes(&lockbytes->ILockBytes_iface, STGM_CREATE|STGM_READWRITE|STGM_TRANSACTED, 0, &stg);
3889     ok(hr==S_OK, "StgCreateDocfileOnILockBytes failed %x\n", hr);
3890 
3891     hr = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, 0, &stm);
3892     ok(hr==S_OK, "IStorage_CreateStream failed %x\n", hr);
3893 
3894     IStream_Release(stm);
3895 
3896     hr = IStorage_Commit(stg, 0);
3897 
3898     IStorage_Release(stg);
3899 
3900     ok(!lockbytes->lock_called, "unexpected call to LockRegion\n");
3901 
3902     lockbytes->locks_supported = LOCK_WRITE|LOCK_EXCLUSIVE|LOCK_ONLYONCE;
3903 
3904     hr = StgCreateDocfileOnILockBytes(&lockbytes->ILockBytes_iface, STGM_CREATE|STGM_READWRITE|STGM_TRANSACTED, 0, &stg);
3905     ok(hr==S_OK, "StgCreateDocfileOnILockBytes failed %x\n", hr);
3906 
3907     hr = IStorage_CreateStream(stg, stmname, STGM_SHARE_EXCLUSIVE|STGM_READWRITE, 0, 0, &stm);
3908     ok(hr==S_OK, "IStorage_CreateStream failed %x\n", hr);
3909 
3910     IStream_Release(stm);
3911 
3912     hr = IStorage_Commit(stg, 0);
3913 
3914     IStorage_Release(stg);
3915 
3916     ok(lockbytes->lock_called, "expected LockRegion to be called\n");
3917 
3918     lockbytes->lock_hr = STG_E_INVALIDFUNCTION;
3919 
3920     hr = StgCreateDocfileOnILockBytes(&lockbytes->ILockBytes_iface, STGM_CREATE|STGM_READWRITE|STGM_TRANSACTED, 0, &stg);
3921     ok(hr==STG_E_INVALIDFUNCTION, "StgCreateDocfileOnILockBytes failed %x\n", hr);
3922 
3923     DeleteTestLockBytes(lockbytes);
3924 }
3925 
START_TEST(storage32)3926 START_TEST(storage32)
3927 {
3928     CHAR temp[MAX_PATH];
3929 
3930     GetTempPathA(MAX_PATH, temp);
3931     if(!GetTempFileNameA(temp, "stg", 0, filenameA))
3932     {
3933         win_skip("Could not create temp file, %u\n", GetLastError());
3934         return;
3935     }
3936     MultiByteToWideChar(CP_ACP, 0, filenameA, -1, filename, MAX_PATH);
3937     DeleteFileA(filenameA);
3938 
3939     test_hglobal_storage_stat();
3940     test_create_storage_modes();
3941     test_stgcreatestorageex();
3942     test_storage_stream();
3943     test_open_storage();
3944     test_storage_suminfo();
3945     test_storage_refcount();
3946     test_streamenum();
3947     test_transact();
3948     test_substorage_share();
3949     test_revert();
3950     test_parent_free();
3951     test_nonroot_transacted();
3952     test_ReadClassStm();
3953     test_access();
3954     test_writeclassstg();
3955     test_readonly();
3956     test_simple();
3957     test_fmtusertypestg();
3958     test_references();
3959     test_copyto();
3960     test_copyto_snbexclusions();
3961     test_copyto_iidexclusions_storage();
3962     test_copyto_iidexclusions_stream();
3963     test_rename();
3964     test_toplevel_stat();
3965     test_substorage_enum();
3966     test_copyto_locking();
3967     test_copyto_recursive();
3968     test_hglobal_storage_creation();
3969     test_convert();
3970     test_direct_swmr();
3971     test_locking();
3972     test_transacted_shared();
3973     test_overwrite();
3974     test_custom_lockbytes();
3975 }
3976