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