xref: /reactos/dll/win32/ole32/storage32.c (revision 62919904)
1 /*
2  * Compound Storage (32 bit version)
3  * Storage implementation
4  *
5  * This file contains the compound file implementation
6  * of the storage interface.
7  *
8  * Copyright 1999 Francis Beaudet
9  * Copyright 1999 Sylvain St-Germain
10  * Copyright 1999 Thuy Nguyen
11  * Copyright 2005 Mike McCormack
12  *
13  * This library is free software; you can redistribute it and/or
14  * modify it under the terms of the GNU Lesser General Public
15  * License as published by the Free Software Foundation; either
16  * version 2.1 of the License, or (at your option) any later version.
17  *
18  * This library is distributed in the hope that it will be useful,
19  * but WITHOUT ANY WARRANTY; without even the implied warranty of
20  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
21  * Lesser General Public License for more details.
22  *
23  * You should have received a copy of the GNU Lesser General Public
24  * License along with this library; if not, write to the Free Software
25  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
26  *
27  * NOTES
28  *  The compound file implementation of IStorage used for create
29  *  and manage substorages and streams within a storage object
30  *  residing in a compound file object.
31  */
32 
33 #include <assert.h>
34 #include <stdarg.h>
35 #include <stdio.h>
36 #include <stdlib.h>
37 #include <string.h>
38 
39 #define COBJMACROS
40 #define NONAMELESSUNION
41 
42 #include "windef.h"
43 #include "winbase.h"
44 #include "winnls.h"
45 #include "winuser.h"
46 #include "wine/unicode.h"
47 #include "wine/debug.h"
48 
49 #include "storage32.h"
50 #include "ole2.h"      /* For Write/ReadClassStm */
51 
52 #include "winreg.h"
53 #include "wine/wingdi16.h"
54 #include "compobj_private.h"
55 
56 WINE_DEFAULT_DEBUG_CHANNEL(storage);
57 
58 
59 /*
60  * These are signatures to detect the type of Document file.
61  */
62 static const BYTE STORAGE_magic[8]    ={0xd0,0xcf,0x11,0xe0,0xa1,0xb1,0x1a,0xe1};
63 static const BYTE STORAGE_oldmagic[8] ={0xd0,0xcf,0x11,0xe0,0x0e,0x11,0xfc,0x0d};
64 
65 extern const IPropertySetStorageVtbl IPropertySetStorage_Vtbl;
66 
67 
68 /****************************************************************************
69  * StorageInternalImpl definitions.
70  *
71  * Definition of the implementation structure for the IStorage interface.
72  * This one implements the IStorage interface for storage that are
73  * inside another storage.
74  */
75 typedef struct StorageInternalImpl
76 {
77   struct StorageBaseImpl base;
78 
79   /*
80    * Entry in the parent's stream tracking list
81    */
82   struct list ParentListEntry;
83 
84   StorageBaseImpl *parentStorage;
85 } StorageInternalImpl;
86 
87 static const IStorageVtbl StorageInternalImpl_Vtbl;
88 static StorageInternalImpl* StorageInternalImpl_Construct(StorageBaseImpl*,DWORD,DirRef);
89 
90 typedef struct TransactedDirEntry
91 {
92   /* If applicable, a reference to the original DirEntry in the transacted
93    * parent. If this is a newly-created entry, DIRENTRY_NULL. */
94   DirRef transactedParentEntry;
95 
96   /* True if this entry is being used. */
97   BOOL inuse;
98 
99   /* True if data is up to date. */
100   BOOL read;
101 
102   /* True if this entry has been modified. */
103   BOOL dirty;
104 
105   /* True if this entry's stream has been modified. */
106   BOOL stream_dirty;
107 
108   /* True if this entry has been deleted in the transacted storage, but the
109    * delete has not yet been committed. */
110   BOOL deleted;
111 
112   /* If this entry's stream has been modified, a reference to where the stream
113    * is stored in the snapshot file. */
114   DirRef stream_entry;
115 
116   /* This directory entry's data, including any changes that have been made. */
117   DirEntry data;
118 
119   /* A reference to the parent of this node. This is only valid while we are
120    * committing changes. */
121   DirRef parent;
122 
123   /* A reference to a newly-created entry in the transacted parent. This is
124    * always equal to transactedParentEntry except when committing changes. */
125   DirRef newTransactedParentEntry;
126 } TransactedDirEntry;
127 
128 
129 /****************************************************************************
130  * Transacted storage object.
131  */
132 typedef struct TransactedSnapshotImpl
133 {
134   struct StorageBaseImpl base;
135 
136   /*
137    * Modified streams are temporarily saved to the scratch file.
138    */
139   StorageBaseImpl *scratch;
140 
141   /* The directory structure is kept here, so that we can track how these
142    * entries relate to those in the parent storage. */
143   TransactedDirEntry *entries;
144   ULONG entries_size;
145   ULONG firstFreeEntry;
146 
147   /*
148    * Changes are committed to the transacted parent.
149    */
150   StorageBaseImpl *transactedParent;
151 
152   /* The transaction signature from when we last committed */
153   ULONG lastTransactionSig;
154 } TransactedSnapshotImpl;
155 
156 static const IStorageVtbl TransactedSnapshotImpl_Vtbl;
157 static HRESULT Storage_ConstructTransacted(StorageBaseImpl*,BOOL,StorageBaseImpl**);
158 
159 typedef struct TransactedSharedImpl
160 {
161   struct StorageBaseImpl base;
162 
163   /*
164    * Snapshot and uncommitted changes go here.
165    */
166   TransactedSnapshotImpl *scratch;
167 
168   /*
169    * Changes are committed to the transacted parent.
170    */
171   StorageBaseImpl *transactedParent;
172 
173   /* The transaction signature from when we last committed */
174   ULONG lastTransactionSig;
175 } TransactedSharedImpl;
176 
177 
178 /****************************************************************************
179  * BlockChainStream definitions.
180  *
181  * The BlockChainStream class is a utility class that is used to create an
182  * abstraction of the big block chains in the storage file.
183  */
184 
185 struct BlockChainRun
186 {
187   /* This represents a range of blocks that happen reside in consecutive sectors. */
188   ULONG firstSector;
189   ULONG firstOffset;
190   ULONG lastOffset;
191 };
192 
193 typedef struct BlockChainBlock
194 {
195   ULONG index;
196   ULONG sector;
197   BOOL  read;
198   BOOL  dirty;
199   BYTE data[MAX_BIG_BLOCK_SIZE];
200 } BlockChainBlock;
201 
202 struct BlockChainStream
203 {
204   StorageImpl* parentStorage;
205   ULONG*       headOfStreamPlaceHolder;
206   DirRef       ownerDirEntry;
207   struct BlockChainRun* indexCache;
208   ULONG        indexCacheLen;
209   ULONG        indexCacheSize;
210   BlockChainBlock cachedBlocks[2];
211   ULONG        blockToEvict;
212   ULONG        tailIndex;
213   ULONG        numBlocks;
214 };
215 
216 /* Returns the number of blocks that comprises this chain.
217  * This is not the size of the stream as the last block may not be full!
218  */
219 static inline ULONG BlockChainStream_GetCount(BlockChainStream* This)
220 {
221   return This->numBlocks;
222 }
223 
224 static BlockChainStream* BlockChainStream_Construct(StorageImpl*,ULONG*,DirRef);
225 static void BlockChainStream_Destroy(BlockChainStream*);
226 static HRESULT BlockChainStream_ReadAt(BlockChainStream*,ULARGE_INTEGER,ULONG,void*,ULONG*);
227 static HRESULT BlockChainStream_WriteAt(BlockChainStream*,ULARGE_INTEGER,ULONG,const void*,ULONG*);
228 static HRESULT BlockChainStream_Flush(BlockChainStream*);
229 static ULARGE_INTEGER BlockChainStream_GetSize(BlockChainStream*);
230 static BOOL BlockChainStream_SetSize(BlockChainStream*,ULARGE_INTEGER);
231 
232 
233 /****************************************************************************
234  * SmallBlockChainStream definitions.
235  *
236  * The SmallBlockChainStream class is a utility class that is used to create an
237  * abstraction of the small block chains in the storage file.
238  */
239 
240 struct SmallBlockChainStream
241 {
242   StorageImpl* parentStorage;
243   DirRef         ownerDirEntry;
244   ULONG*         headOfStreamPlaceHolder;
245 };
246 
247 static SmallBlockChainStream* SmallBlockChainStream_Construct(StorageImpl*,ULONG*,DirRef);
248 static void SmallBlockChainStream_Destroy(SmallBlockChainStream*);
249 static HRESULT SmallBlockChainStream_ReadAt(SmallBlockChainStream*,ULARGE_INTEGER,ULONG,void*,ULONG*);
250 static HRESULT SmallBlockChainStream_WriteAt(SmallBlockChainStream*,ULARGE_INTEGER,ULONG,const void*,ULONG*);
251 static ULARGE_INTEGER SmallBlockChainStream_GetSize(SmallBlockChainStream*);
252 static BOOL SmallBlockChainStream_SetSize(SmallBlockChainStream*,ULARGE_INTEGER);
253 
254 
255 /************************************************************************
256  * STGM Functions
257  ***********************************************************************/
258 
259 /************************************************************************
260  * This method validates an STGM parameter that can contain the values below
261  *
262  * The stgm modes in 0x0000ffff are not bit masks, but distinct 4 bit values.
263  * The stgm values contained in 0xffff0000 are bitmasks.
264  *
265  * STGM_DIRECT               0x00000000
266  * STGM_TRANSACTED           0x00010000
267  * STGM_SIMPLE               0x08000000
268  *
269  * STGM_READ                 0x00000000
270  * STGM_WRITE                0x00000001
271  * STGM_READWRITE            0x00000002
272  *
273  * STGM_SHARE_DENY_NONE      0x00000040
274  * STGM_SHARE_DENY_READ      0x00000030
275  * STGM_SHARE_DENY_WRITE     0x00000020
276  * STGM_SHARE_EXCLUSIVE      0x00000010
277  *
278  * STGM_PRIORITY             0x00040000
279  * STGM_DELETEONRELEASE      0x04000000
280  *
281  * STGM_CREATE               0x00001000
282  * STGM_CONVERT              0x00020000
283  * STGM_FAILIFTHERE          0x00000000
284  *
285  * STGM_NOSCRATCH            0x00100000
286  * STGM_NOSNAPSHOT           0x00200000
287  */
288 static HRESULT validateSTGM(DWORD stgm)
289 {
290   DWORD access = STGM_ACCESS_MODE(stgm);
291   DWORD share  = STGM_SHARE_MODE(stgm);
292   DWORD create = STGM_CREATE_MODE(stgm);
293 
294   if (stgm&~STGM_KNOWN_FLAGS)
295   {
296     ERR("unknown flags %08x\n", stgm);
297     return E_FAIL;
298   }
299 
300   switch (access)
301   {
302   case STGM_READ:
303   case STGM_WRITE:
304   case STGM_READWRITE:
305     break;
306   default:
307     return E_FAIL;
308   }
309 
310   switch (share)
311   {
312   case STGM_SHARE_DENY_NONE:
313   case STGM_SHARE_DENY_READ:
314   case STGM_SHARE_DENY_WRITE:
315   case STGM_SHARE_EXCLUSIVE:
316     break;
317   case 0:
318     if (!(stgm & STGM_TRANSACTED))
319       return E_FAIL;
320     break;
321   default:
322     return E_FAIL;
323   }
324 
325   switch (create)
326   {
327   case STGM_CREATE:
328   case STGM_FAILIFTHERE:
329     break;
330   default:
331     return E_FAIL;
332   }
333 
334   /*
335    * STGM_DIRECT | STGM_TRANSACTED | STGM_SIMPLE
336    */
337   if ( (stgm & STGM_TRANSACTED) && (stgm & STGM_SIMPLE) )
338       return E_FAIL;
339 
340   /*
341    * STGM_CREATE | STGM_CONVERT
342    * if both are false, STGM_FAILIFTHERE is set to TRUE
343    */
344   if ( create == STGM_CREATE && (stgm & STGM_CONVERT) )
345     return E_FAIL;
346 
347   /*
348    * STGM_NOSCRATCH requires STGM_TRANSACTED
349    */
350   if ( (stgm & STGM_NOSCRATCH) && !(stgm & STGM_TRANSACTED) )
351     return E_FAIL;
352 
353   /*
354    * STGM_NOSNAPSHOT requires STGM_TRANSACTED and
355    * not STGM_SHARE_EXCLUSIVE or STGM_SHARE_DENY_WRITE`
356    */
357   if ( (stgm & STGM_NOSNAPSHOT) &&
358         (!(stgm & STGM_TRANSACTED) ||
359          share == STGM_SHARE_EXCLUSIVE ||
360          share == STGM_SHARE_DENY_WRITE) )
361     return E_FAIL;
362 
363   return S_OK;
364 }
365 
366 /************************************************************************
367  *      GetShareModeFromSTGM
368  *
369  * This method will return a share mode flag from a STGM value.
370  * The STGM value is assumed valid.
371  */
372 static DWORD GetShareModeFromSTGM(DWORD stgm)
373 {
374   switch (STGM_SHARE_MODE(stgm))
375   {
376   case 0:
377     assert(stgm & STGM_TRANSACTED);
378     /* fall-through */
379   case STGM_SHARE_DENY_NONE:
380     return FILE_SHARE_READ | FILE_SHARE_WRITE;
381   case STGM_SHARE_DENY_READ:
382     return FILE_SHARE_WRITE;
383   case STGM_SHARE_DENY_WRITE:
384   case STGM_SHARE_EXCLUSIVE:
385     return FILE_SHARE_READ;
386   }
387   ERR("Invalid share mode!\n");
388   assert(0);
389   return 0;
390 }
391 
392 /************************************************************************
393  *      GetAccessModeFromSTGM
394  *
395  * This method will return an access mode flag from a STGM value.
396  * The STGM value is assumed valid.
397  */
398 static DWORD GetAccessModeFromSTGM(DWORD stgm)
399 {
400   switch (STGM_ACCESS_MODE(stgm))
401   {
402   case STGM_READ:
403     return GENERIC_READ;
404   case STGM_WRITE:
405   case STGM_READWRITE:
406     return GENERIC_READ | GENERIC_WRITE;
407   }
408   ERR("Invalid access mode!\n");
409   assert(0);
410   return 0;
411 }
412 
413 /************************************************************************
414  *      GetCreationModeFromSTGM
415  *
416  * This method will return a creation mode flag from a STGM value.
417  * The STGM value is assumed valid.
418  */
419 static DWORD GetCreationModeFromSTGM(DWORD stgm)
420 {
421   switch(STGM_CREATE_MODE(stgm))
422   {
423   case STGM_CREATE:
424     return CREATE_ALWAYS;
425   case STGM_CONVERT:
426     FIXME("STGM_CONVERT not implemented!\n");
427     return CREATE_NEW;
428   case STGM_FAILIFTHERE:
429     return CREATE_NEW;
430   }
431   ERR("Invalid create mode!\n");
432   assert(0);
433   return 0;
434 }
435 
436 
437 /************************************************************************
438  * IDirectWriterLock implementation
439  ***********************************************************************/
440 
441 static inline StorageBaseImpl *impl_from_IDirectWriterLock( IDirectWriterLock *iface )
442 {
443     return CONTAINING_RECORD(iface, StorageBaseImpl, IDirectWriterLock_iface);
444 }
445 
446 static HRESULT WINAPI directwriterlock_QueryInterface(IDirectWriterLock *iface, REFIID riid, void **obj)
447 {
448   StorageBaseImpl *This = impl_from_IDirectWriterLock(iface);
449   return IStorage_QueryInterface(&This->IStorage_iface, riid, obj);
450 }
451 
452 static ULONG WINAPI directwriterlock_AddRef(IDirectWriterLock *iface)
453 {
454   StorageBaseImpl *This = impl_from_IDirectWriterLock(iface);
455   return IStorage_AddRef(&This->IStorage_iface);
456 }
457 
458 static ULONG WINAPI directwriterlock_Release(IDirectWriterLock *iface)
459 {
460   StorageBaseImpl *This = impl_from_IDirectWriterLock(iface);
461   return IStorage_Release(&This->IStorage_iface);
462 }
463 
464 static HRESULT WINAPI directwriterlock_WaitForWriteAccess(IDirectWriterLock *iface, DWORD timeout)
465 {
466   StorageBaseImpl *This = impl_from_IDirectWriterLock(iface);
467   FIXME("(%p)->(%d): stub\n", This, timeout);
468   return E_NOTIMPL;
469 }
470 
471 static HRESULT WINAPI directwriterlock_ReleaseWriteAccess(IDirectWriterLock *iface)
472 {
473   StorageBaseImpl *This = impl_from_IDirectWriterLock(iface);
474   FIXME("(%p): stub\n", This);
475   return E_NOTIMPL;
476 }
477 
478 static HRESULT WINAPI directwriterlock_HaveWriteAccess(IDirectWriterLock *iface)
479 {
480   StorageBaseImpl *This = impl_from_IDirectWriterLock(iface);
481   FIXME("(%p): stub\n", This);
482   return E_NOTIMPL;
483 }
484 
485 static const IDirectWriterLockVtbl DirectWriterLockVtbl =
486 {
487   directwriterlock_QueryInterface,
488   directwriterlock_AddRef,
489   directwriterlock_Release,
490   directwriterlock_WaitForWriteAccess,
491   directwriterlock_ReleaseWriteAccess,
492   directwriterlock_HaveWriteAccess
493 };
494 
495 
496 /************************************************************************
497  * StorageBaseImpl implementation : Tree helper functions
498  ***********************************************************************/
499 
500 /****************************************************************************
501  *
502  * Internal Method
503  *
504  * Case insensitive comparison of DirEntry.name by first considering
505  * their size.
506  *
507  * Returns <0 when name1 < name2
508  *         >0 when name1 > name2
509  *          0 when name1 == name2
510  */
511 static LONG entryNameCmp(
512     const OLECHAR *name1,
513     const OLECHAR *name2)
514 {
515   LONG diff      = lstrlenW(name1) - lstrlenW(name2);
516 
517   while (diff == 0 && *name1 != 0)
518   {
519     /*
520      * We compare the string themselves only when they are of the same length
521      */
522     diff = toupperW(*name1++) - toupperW(*name2++);
523   }
524 
525   return diff;
526 }
527 
528 /****************************************************************************
529  *
530  * Internal Method
531  *
532  * Find and read the element of a storage with the given name.
533  */
534 static DirRef findElement(StorageBaseImpl *storage, DirRef storageEntry,
535     const OLECHAR *name, DirEntry *data)
536 {
537   DirRef currentEntry;
538 
539   /* Read the storage entry to find the root of the tree. */
540   StorageBaseImpl_ReadDirEntry(storage, storageEntry, data);
541 
542   currentEntry = data->dirRootEntry;
543 
544   while (currentEntry != DIRENTRY_NULL)
545   {
546     LONG cmp;
547 
548     StorageBaseImpl_ReadDirEntry(storage, currentEntry, data);
549 
550     cmp = entryNameCmp(name, data->name);
551 
552     if (cmp == 0)
553       /* found it */
554       break;
555 
556     else if (cmp < 0)
557       currentEntry = data->leftChild;
558 
559     else if (cmp > 0)
560       currentEntry = data->rightChild;
561   }
562 
563   return currentEntry;
564 }
565 
566 /****************************************************************************
567  *
568  * Internal Method
569  *
570  * Find and read the binary tree parent of the element with the given name.
571  *
572  * If there is no such element, find a place where it could be inserted and
573  * return STG_E_FILENOTFOUND.
574  */
575 static HRESULT findTreeParent(StorageBaseImpl *storage, DirRef storageEntry,
576     const OLECHAR *childName, DirEntry *parentData, DirRef *parentEntry,
577     ULONG *relation)
578 {
579   DirRef childEntry;
580   DirEntry childData;
581 
582   /* Read the storage entry to find the root of the tree. */
583   StorageBaseImpl_ReadDirEntry(storage, storageEntry, parentData);
584 
585   *parentEntry = storageEntry;
586   *relation = DIRENTRY_RELATION_DIR;
587 
588   childEntry = parentData->dirRootEntry;
589 
590   while (childEntry != DIRENTRY_NULL)
591   {
592     LONG cmp;
593 
594     StorageBaseImpl_ReadDirEntry(storage, childEntry, &childData);
595 
596     cmp = entryNameCmp(childName, childData.name);
597 
598     if (cmp == 0)
599       /* found it */
600       break;
601 
602     else if (cmp < 0)
603     {
604       *parentData = childData;
605       *parentEntry = childEntry;
606       *relation = DIRENTRY_RELATION_PREVIOUS;
607 
608       childEntry = parentData->leftChild;
609     }
610 
611     else if (cmp > 0)
612     {
613       *parentData = childData;
614       *parentEntry = childEntry;
615       *relation = DIRENTRY_RELATION_NEXT;
616 
617       childEntry = parentData->rightChild;
618     }
619   }
620 
621   if (childEntry == DIRENTRY_NULL)
622     return STG_E_FILENOTFOUND;
623   else
624     return S_OK;
625 }
626 
627 static void setEntryLink(DirEntry *entry, ULONG relation, DirRef new_target)
628 {
629   switch (relation)
630   {
631     case DIRENTRY_RELATION_PREVIOUS:
632       entry->leftChild = new_target;
633       break;
634     case DIRENTRY_RELATION_NEXT:
635       entry->rightChild = new_target;
636       break;
637     case DIRENTRY_RELATION_DIR:
638       entry->dirRootEntry = new_target;
639       break;
640     default:
641       assert(0);
642   }
643 }
644 
645 /****************************************************************************
646  *
647  * Internal Method
648  *
649  * Add a directory entry to a storage
650  */
651 static HRESULT insertIntoTree(
652   StorageBaseImpl *This,
653   DirRef        parentStorageIndex,
654   DirRef        newEntryIndex)
655 {
656   DirEntry currentEntry;
657   DirEntry newEntry;
658 
659   /*
660    * Read the inserted entry
661    */
662   StorageBaseImpl_ReadDirEntry(This,
663                                newEntryIndex,
664                                &newEntry);
665 
666   /*
667    * Read the storage entry
668    */
669   StorageBaseImpl_ReadDirEntry(This,
670                                parentStorageIndex,
671                                &currentEntry);
672 
673   if (currentEntry.dirRootEntry != DIRENTRY_NULL)
674   {
675     /*
676      * The root storage contains some element, therefore, start the research
677      * for the appropriate location.
678      */
679     BOOL found = FALSE;
680     DirRef current, next, previous, currentEntryId;
681 
682     /*
683      * Keep a reference to the root of the storage's element tree
684      */
685     currentEntryId = currentEntry.dirRootEntry;
686 
687     /*
688      * Read
689      */
690     StorageBaseImpl_ReadDirEntry(This,
691                                  currentEntry.dirRootEntry,
692                                  &currentEntry);
693 
694     previous = currentEntry.leftChild;
695     next     = currentEntry.rightChild;
696     current  = currentEntryId;
697 
698     while (!found)
699     {
700       LONG diff = entryNameCmp( newEntry.name, currentEntry.name);
701 
702       if (diff < 0)
703       {
704         if (previous != DIRENTRY_NULL)
705         {
706           StorageBaseImpl_ReadDirEntry(This,
707                                        previous,
708                                        &currentEntry);
709           current = previous;
710         }
711         else
712         {
713           currentEntry.leftChild = newEntryIndex;
714           StorageBaseImpl_WriteDirEntry(This,
715                                         current,
716                                         &currentEntry);
717           found = TRUE;
718         }
719       }
720       else if (diff > 0)
721       {
722         if (next != DIRENTRY_NULL)
723         {
724           StorageBaseImpl_ReadDirEntry(This,
725                                        next,
726                                        &currentEntry);
727           current = next;
728         }
729         else
730         {
731           currentEntry.rightChild = newEntryIndex;
732           StorageBaseImpl_WriteDirEntry(This,
733                                         current,
734                                         &currentEntry);
735           found = TRUE;
736         }
737       }
738       else
739       {
740 	/*
741 	 * Trying to insert an item with the same name in the
742 	 * subtree structure.
743 	 */
744 	return STG_E_FILEALREADYEXISTS;
745       }
746 
747       previous = currentEntry.leftChild;
748       next     = currentEntry.rightChild;
749     }
750   }
751   else
752   {
753     /*
754      * The storage is empty, make the new entry the root of its element tree
755      */
756     currentEntry.dirRootEntry = newEntryIndex;
757     StorageBaseImpl_WriteDirEntry(This,
758                                   parentStorageIndex,
759                                   &currentEntry);
760   }
761 
762   return S_OK;
763 }
764 
765 /*************************************************************************
766  *
767  * Internal Method
768  *
769  * This method removes a directory entry from its parent storage tree without
770  * freeing any resources attached to it.
771  */
772 static HRESULT removeFromTree(
773   StorageBaseImpl *This,
774   DirRef        parentStorageIndex,
775   DirRef        deletedIndex)
776 {
777   DirEntry   entryToDelete;
778   DirEntry   parentEntry;
779   DirRef parentEntryRef;
780   ULONG typeOfRelation;
781   HRESULT hr;
782 
783   hr = StorageBaseImpl_ReadDirEntry(This, deletedIndex, &entryToDelete);
784 
785   if (hr != S_OK)
786     return hr;
787 
788   /*
789    * Find the element that links to the one we want to delete.
790    */
791   hr = findTreeParent(This, parentStorageIndex, entryToDelete.name,
792     &parentEntry, &parentEntryRef, &typeOfRelation);
793 
794   if (hr != S_OK)
795     return hr;
796 
797   if (entryToDelete.leftChild != DIRENTRY_NULL)
798   {
799     /*
800      * Replace the deleted entry with its left child
801      */
802     setEntryLink(&parentEntry, typeOfRelation, entryToDelete.leftChild);
803 
804     hr = StorageBaseImpl_WriteDirEntry(
805             This,
806             parentEntryRef,
807             &parentEntry);
808     if(FAILED(hr))
809     {
810       return hr;
811     }
812 
813     if (entryToDelete.rightChild != DIRENTRY_NULL)
814     {
815       /*
816        * We need to reinsert the right child somewhere. We already know it and
817        * its children are greater than everything in the left tree, so we
818        * insert it at the rightmost point in the left tree.
819        */
820       DirRef newRightChildParent = entryToDelete.leftChild;
821       DirEntry newRightChildParentEntry;
822 
823       do
824       {
825         hr = StorageBaseImpl_ReadDirEntry(
826                 This,
827                 newRightChildParent,
828                 &newRightChildParentEntry);
829         if (FAILED(hr))
830         {
831           return hr;
832         }
833 
834         if (newRightChildParentEntry.rightChild != DIRENTRY_NULL)
835           newRightChildParent = newRightChildParentEntry.rightChild;
836       } while (newRightChildParentEntry.rightChild != DIRENTRY_NULL);
837 
838       newRightChildParentEntry.rightChild = entryToDelete.rightChild;
839 
840       hr = StorageBaseImpl_WriteDirEntry(
841               This,
842               newRightChildParent,
843               &newRightChildParentEntry);
844       if (FAILED(hr))
845       {
846         return hr;
847       }
848     }
849   }
850   else
851   {
852     /*
853      * Replace the deleted entry with its right child
854      */
855     setEntryLink(&parentEntry, typeOfRelation, entryToDelete.rightChild);
856 
857     hr = StorageBaseImpl_WriteDirEntry(
858             This,
859             parentEntryRef,
860             &parentEntry);
861     if(FAILED(hr))
862     {
863       return hr;
864     }
865   }
866 
867   return hr;
868 }
869 
870 
871 /************************************************************************
872  * IEnumSTATSTGImpl implementation for StorageBaseImpl_EnumElements
873  ***********************************************************************/
874 
875 /*
876  * IEnumSTATSTGImpl definitions.
877  *
878  * Definition of the implementation structure for the IEnumSTATSTGImpl interface.
879  * This class allows iterating through the content of a storage and finding
880  * specific items inside it.
881  */
882 struct IEnumSTATSTGImpl
883 {
884   IEnumSTATSTG   IEnumSTATSTG_iface;
885 
886   LONG           ref;                   /* Reference count */
887   StorageBaseImpl* parentStorage;         /* Reference to the parent storage */
888   DirRef         storageDirEntry;     /* Directory entry of the storage to enumerate */
889 
890   WCHAR	         name[DIRENTRY_NAME_MAX_LEN]; /* The most recent name visited */
891 };
892 
893 static inline IEnumSTATSTGImpl *impl_from_IEnumSTATSTG(IEnumSTATSTG *iface)
894 {
895   return CONTAINING_RECORD(iface, IEnumSTATSTGImpl, IEnumSTATSTG_iface);
896 }
897 
898 static void IEnumSTATSTGImpl_Destroy(IEnumSTATSTGImpl* This)
899 {
900   IStorage_Release(&This->parentStorage->IStorage_iface);
901   HeapFree(GetProcessHeap(), 0, This);
902 }
903 
904 static HRESULT WINAPI IEnumSTATSTGImpl_QueryInterface(
905   IEnumSTATSTG*     iface,
906   REFIID            riid,
907   void**            ppvObject)
908 {
909   IEnumSTATSTGImpl* const This = impl_from_IEnumSTATSTG(iface);
910 
911   TRACE("%p,%s,%p\n", iface, debugstr_guid(riid), ppvObject);
912 
913   if (ppvObject==0)
914     return E_INVALIDARG;
915 
916   *ppvObject = 0;
917 
918   if (IsEqualGUID(&IID_IUnknown, riid) ||
919       IsEqualGUID(&IID_IEnumSTATSTG, riid))
920   {
921     *ppvObject = &This->IEnumSTATSTG_iface;
922     IEnumSTATSTG_AddRef(&This->IEnumSTATSTG_iface);
923     TRACE("<-- %p\n", *ppvObject);
924     return S_OK;
925   }
926 
927   TRACE("<-- E_NOINTERFACE\n");
928   return E_NOINTERFACE;
929 }
930 
931 static ULONG   WINAPI IEnumSTATSTGImpl_AddRef(
932   IEnumSTATSTG* iface)
933 {
934   IEnumSTATSTGImpl* const This = impl_from_IEnumSTATSTG(iface);
935   return InterlockedIncrement(&This->ref);
936 }
937 
938 static ULONG   WINAPI IEnumSTATSTGImpl_Release(
939   IEnumSTATSTG* iface)
940 {
941   IEnumSTATSTGImpl* const This = impl_from_IEnumSTATSTG(iface);
942 
943   ULONG newRef;
944 
945   newRef = InterlockedDecrement(&This->ref);
946 
947   if (newRef==0)
948   {
949     IEnumSTATSTGImpl_Destroy(This);
950   }
951 
952   return newRef;
953 }
954 
955 static HRESULT IEnumSTATSTGImpl_GetNextRef(
956   IEnumSTATSTGImpl* This,
957   DirRef *ref)
958 {
959   DirRef result = DIRENTRY_NULL;
960   DirRef searchNode;
961   DirEntry entry;
962   HRESULT hr;
963   WCHAR result_name[DIRENTRY_NAME_MAX_LEN];
964 
965   TRACE("%p,%p\n", This, ref);
966 
967   hr = StorageBaseImpl_ReadDirEntry(This->parentStorage,
968     This->parentStorage->storageDirEntry, &entry);
969   searchNode = entry.dirRootEntry;
970 
971   while (SUCCEEDED(hr) && searchNode != DIRENTRY_NULL)
972   {
973     hr = StorageBaseImpl_ReadDirEntry(This->parentStorage, searchNode, &entry);
974 
975     if (SUCCEEDED(hr))
976     {
977       LONG diff = entryNameCmp( entry.name, This->name);
978 
979       if (diff <= 0)
980       {
981         searchNode = entry.rightChild;
982       }
983       else
984       {
985         result = searchNode;
986         memcpy(result_name, entry.name, sizeof(result_name));
987         searchNode = entry.leftChild;
988       }
989     }
990   }
991 
992   if (SUCCEEDED(hr))
993   {
994     *ref = result;
995     if (result != DIRENTRY_NULL)
996       memcpy(This->name, result_name, sizeof(result_name));
997   }
998 
999   TRACE("<-- %08x\n", hr);
1000   return hr;
1001 }
1002 
1003 static HRESULT WINAPI IEnumSTATSTGImpl_Next(
1004   IEnumSTATSTG* iface,
1005   ULONG             celt,
1006   STATSTG*          rgelt,
1007   ULONG*            pceltFetched)
1008 {
1009   IEnumSTATSTGImpl* const This = impl_from_IEnumSTATSTG(iface);
1010 
1011   DirEntry    currentEntry;
1012   STATSTG*    currentReturnStruct = rgelt;
1013   ULONG       objectFetched       = 0;
1014   DirRef      currentSearchNode;
1015   HRESULT     hr=S_OK;
1016 
1017   TRACE("%p,%u,%p,%p\n", iface, celt, rgelt, pceltFetched);
1018 
1019   if ( (rgelt==0) || ( (celt!=1) && (pceltFetched==0) ) )
1020     return E_INVALIDARG;
1021 
1022   if (This->parentStorage->reverted)
1023   {
1024     TRACE("<-- STG_E_REVERTED\n");
1025     return STG_E_REVERTED;
1026   }
1027 
1028   /*
1029    * To avoid the special case, get another pointer to a ULONG value if
1030    * the caller didn't supply one.
1031    */
1032   if (pceltFetched==0)
1033     pceltFetched = &objectFetched;
1034 
1035   /*
1036    * Start the iteration, we will iterate until we hit the end of the
1037    * linked list or until we hit the number of items to iterate through
1038    */
1039   *pceltFetched = 0;
1040 
1041   while ( *pceltFetched < celt )
1042   {
1043     hr = IEnumSTATSTGImpl_GetNextRef(This, &currentSearchNode);
1044 
1045     if (FAILED(hr) || currentSearchNode == DIRENTRY_NULL)
1046     {
1047       memset(currentReturnStruct, 0, sizeof(*currentReturnStruct));
1048       break;
1049     }
1050 
1051     /*
1052      * Read the entry from the storage.
1053      */
1054     hr = StorageBaseImpl_ReadDirEntry(This->parentStorage,
1055       currentSearchNode,
1056       &currentEntry);
1057     if (FAILED(hr)) break;
1058 
1059     /*
1060      * Copy the information to the return buffer.
1061      */
1062     StorageUtl_CopyDirEntryToSTATSTG(This->parentStorage,
1063       currentReturnStruct,
1064       &currentEntry,
1065       STATFLAG_DEFAULT);
1066 
1067     /*
1068      * Step to the next item in the iteration
1069      */
1070     (*pceltFetched)++;
1071     currentReturnStruct++;
1072   }
1073 
1074   if (SUCCEEDED(hr) && *pceltFetched != celt)
1075     hr = S_FALSE;
1076 
1077   TRACE("<-- %08x (asked %u, got %u)\n", hr, celt, *pceltFetched);
1078   return hr;
1079 }
1080 
1081 
1082 static HRESULT WINAPI IEnumSTATSTGImpl_Skip(
1083   IEnumSTATSTG* iface,
1084   ULONG             celt)
1085 {
1086   IEnumSTATSTGImpl* const This = impl_from_IEnumSTATSTG(iface);
1087 
1088   ULONG       objectFetched = 0;
1089   DirRef      currentSearchNode;
1090   HRESULT     hr=S_OK;
1091 
1092   TRACE("%p,%u\n", iface, celt);
1093 
1094   if (This->parentStorage->reverted)
1095   {
1096     TRACE("<-- STG_E_REVERTED\n");
1097     return STG_E_REVERTED;
1098   }
1099 
1100   while ( (objectFetched < celt) )
1101   {
1102     hr = IEnumSTATSTGImpl_GetNextRef(This, &currentSearchNode);
1103 
1104     if (FAILED(hr) || currentSearchNode == DIRENTRY_NULL)
1105       break;
1106 
1107     objectFetched++;
1108   }
1109 
1110   if (SUCCEEDED(hr) && objectFetched != celt)
1111     return S_FALSE;
1112 
1113   TRACE("<-- %08x\n", hr);
1114   return hr;
1115 }
1116 
1117 static HRESULT WINAPI IEnumSTATSTGImpl_Reset(
1118   IEnumSTATSTG* iface)
1119 {
1120   IEnumSTATSTGImpl* const This = impl_from_IEnumSTATSTG(iface);
1121 
1122   TRACE("%p\n", iface);
1123 
1124   if (This->parentStorage->reverted)
1125   {
1126     TRACE("<-- STG_E_REVERTED\n");
1127     return STG_E_REVERTED;
1128   }
1129 
1130   This->name[0] = 0;
1131 
1132   return S_OK;
1133 }
1134 
1135 static IEnumSTATSTGImpl* IEnumSTATSTGImpl_Construct(StorageBaseImpl*,DirRef);
1136 
1137 static HRESULT WINAPI IEnumSTATSTGImpl_Clone(
1138   IEnumSTATSTG* iface,
1139   IEnumSTATSTG**    ppenum)
1140 {
1141   IEnumSTATSTGImpl* const This = impl_from_IEnumSTATSTG(iface);
1142   IEnumSTATSTGImpl* newClone;
1143 
1144   TRACE("%p,%p\n", iface, ppenum);
1145 
1146   if (This->parentStorage->reverted)
1147   {
1148     TRACE("<-- STG_E_REVERTED\n");
1149     return STG_E_REVERTED;
1150   }
1151 
1152   if (ppenum==0)
1153     return E_INVALIDARG;
1154 
1155   newClone = IEnumSTATSTGImpl_Construct(This->parentStorage,
1156                This->storageDirEntry);
1157   if (!newClone)
1158   {
1159     *ppenum = NULL;
1160     return E_OUTOFMEMORY;
1161   }
1162 
1163   /*
1164    * The new clone enumeration must point to the same current node as
1165    * the old one.
1166    */
1167   memcpy(newClone->name, This->name, sizeof(newClone->name));
1168 
1169   *ppenum = &newClone->IEnumSTATSTG_iface;
1170 
1171   return S_OK;
1172 }
1173 
1174 /*
1175  * Virtual function table for the IEnumSTATSTGImpl class.
1176  */
1177 static const IEnumSTATSTGVtbl IEnumSTATSTGImpl_Vtbl =
1178 {
1179     IEnumSTATSTGImpl_QueryInterface,
1180     IEnumSTATSTGImpl_AddRef,
1181     IEnumSTATSTGImpl_Release,
1182     IEnumSTATSTGImpl_Next,
1183     IEnumSTATSTGImpl_Skip,
1184     IEnumSTATSTGImpl_Reset,
1185     IEnumSTATSTGImpl_Clone
1186 };
1187 
1188 static IEnumSTATSTGImpl* IEnumSTATSTGImpl_Construct(
1189   StorageBaseImpl* parentStorage,
1190   DirRef         storageDirEntry)
1191 {
1192   IEnumSTATSTGImpl* newEnumeration;
1193 
1194   newEnumeration = HeapAlloc(GetProcessHeap(), 0, sizeof(IEnumSTATSTGImpl));
1195 
1196   if (newEnumeration)
1197   {
1198     newEnumeration->IEnumSTATSTG_iface.lpVtbl = &IEnumSTATSTGImpl_Vtbl;
1199     newEnumeration->ref = 1;
1200     newEnumeration->name[0] = 0;
1201 
1202     /*
1203      * We want to nail-down the reference to the storage in case the
1204      * enumeration out-lives the storage in the client application.
1205      */
1206     newEnumeration->parentStorage = parentStorage;
1207     IStorage_AddRef(&newEnumeration->parentStorage->IStorage_iface);
1208 
1209     newEnumeration->storageDirEntry = storageDirEntry;
1210   }
1211 
1212   return newEnumeration;
1213 }
1214 
1215 
1216 /************************************************************************
1217  * StorageBaseImpl implementation
1218  ***********************************************************************/
1219 
1220 static inline StorageBaseImpl *impl_from_IStorage( IStorage *iface )
1221 {
1222     return CONTAINING_RECORD(iface, StorageBaseImpl, IStorage_iface);
1223 }
1224 
1225 /************************************************************************
1226  * StorageBaseImpl_QueryInterface (IUnknown)
1227  *
1228  * This method implements the common QueryInterface for all IStorage
1229  * implementations contained in this file.
1230  *
1231  * See Windows documentation for more details on IUnknown methods.
1232  */
1233 static HRESULT WINAPI StorageBaseImpl_QueryInterface(
1234   IStorage*        iface,
1235   REFIID             riid,
1236   void**             ppvObject)
1237 {
1238   StorageBaseImpl *This = impl_from_IStorage(iface);
1239 
1240   TRACE("%p,%s,%p\n", iface, debugstr_guid(riid), ppvObject);
1241 
1242   if (!ppvObject)
1243     return E_INVALIDARG;
1244 
1245   *ppvObject = 0;
1246 
1247   if (IsEqualGUID(&IID_IUnknown, riid) ||
1248       IsEqualGUID(&IID_IStorage, riid))
1249   {
1250     *ppvObject = &This->IStorage_iface;
1251   }
1252   else if (IsEqualGUID(&IID_IPropertySetStorage, riid))
1253   {
1254     *ppvObject = &This->IPropertySetStorage_iface;
1255   }
1256   /* locking interface is reported for writer only */
1257   else if (IsEqualGUID(&IID_IDirectWriterLock, riid) && This->lockingrole == SWMR_Writer)
1258   {
1259     *ppvObject = &This->IDirectWriterLock_iface;
1260   }
1261   else
1262   {
1263     TRACE("<-- E_NOINTERFACE\n");
1264     return E_NOINTERFACE;
1265   }
1266 
1267   IStorage_AddRef(iface);
1268   TRACE("<-- %p\n", *ppvObject);
1269   return S_OK;
1270 }
1271 
1272 /************************************************************************
1273  * StorageBaseImpl_AddRef (IUnknown)
1274  *
1275  * This method implements the common AddRef for all IStorage
1276  * implementations contained in this file.
1277  *
1278  * See Windows documentation for more details on IUnknown methods.
1279  */
1280 static ULONG WINAPI StorageBaseImpl_AddRef(
1281             IStorage* iface)
1282 {
1283   StorageBaseImpl *This = impl_from_IStorage(iface);
1284   ULONG ref = InterlockedIncrement(&This->ref);
1285 
1286   TRACE("(%p) AddRef to %d\n", This, ref);
1287 
1288   return ref;
1289 }
1290 
1291 /************************************************************************
1292  * StorageBaseImpl_Release (IUnknown)
1293  *
1294  * This method implements the common Release for all IStorage
1295  * implementations contained in this file.
1296  *
1297  * See Windows documentation for more details on IUnknown methods.
1298  */
1299 static ULONG WINAPI StorageBaseImpl_Release(
1300       IStorage* iface)
1301 {
1302   StorageBaseImpl *This = impl_from_IStorage(iface);
1303 
1304   ULONG ref = InterlockedDecrement(&This->ref);
1305 
1306   TRACE("(%p) ReleaseRef to %d\n", This, ref);
1307 
1308   if (ref == 0)
1309   {
1310     /*
1311      * Since we are using a system of base-classes, we want to call the
1312      * destructor of the appropriate derived class. To do this, we are
1313      * using virtual functions to implement the destructor.
1314      */
1315     StorageBaseImpl_Destroy(This);
1316   }
1317 
1318   return ref;
1319 }
1320 
1321 static HRESULT StorageBaseImpl_CopyStorageEntryTo(StorageBaseImpl *This,
1322     DirRef srcEntry, BOOL skip_storage, BOOL skip_stream,
1323     SNB snbExclude, IStorage *pstgDest);
1324 
1325 static HRESULT StorageBaseImpl_CopyChildEntryTo(StorageBaseImpl *This,
1326     DirRef srcEntry, BOOL skip_storage, BOOL skip_stream,
1327     SNB snbExclude, IStorage *pstgDest)
1328 {
1329   DirEntry data;
1330   HRESULT hr;
1331   BOOL skip = FALSE;
1332   IStorage *pstgTmp;
1333   IStream *pstrChild, *pstrTmp;
1334   STATSTG strStat;
1335 
1336   if (srcEntry == DIRENTRY_NULL)
1337     return S_OK;
1338 
1339   hr = StorageBaseImpl_ReadDirEntry( This, srcEntry, &data );
1340 
1341   if (FAILED(hr))
1342     return hr;
1343 
1344   if ( snbExclude )
1345   {
1346     WCHAR **snb = snbExclude;
1347 
1348     while ( *snb != NULL && !skip )
1349     {
1350       if ( lstrcmpW(data.name, *snb) == 0 )
1351         skip = TRUE;
1352       ++snb;
1353     }
1354   }
1355 
1356   if (!skip)
1357   {
1358     if (data.stgType == STGTY_STORAGE && !skip_storage)
1359     {
1360       /*
1361        * create a new storage in destination storage
1362        */
1363       hr = IStorage_CreateStorage( pstgDest, data.name,
1364                                    STGM_FAILIFTHERE|STGM_WRITE|STGM_SHARE_EXCLUSIVE,
1365                                    0, 0,
1366                                    &pstgTmp );
1367 
1368       /*
1369        * if it already exist, don't create a new one use this one
1370        */
1371       if (hr == STG_E_FILEALREADYEXISTS)
1372       {
1373         hr = IStorage_OpenStorage( pstgDest, data.name, NULL,
1374                                    STGM_WRITE|STGM_SHARE_EXCLUSIVE,
1375                                    NULL, 0, &pstgTmp );
1376       }
1377 
1378       if (SUCCEEDED(hr))
1379       {
1380         hr = StorageBaseImpl_CopyStorageEntryTo( This, srcEntry, skip_storage,
1381                                                  skip_stream, NULL, pstgTmp );
1382 
1383         IStorage_Release(pstgTmp);
1384       }
1385     }
1386     else if (data.stgType == STGTY_STREAM && !skip_stream)
1387     {
1388       /*
1389        * create a new stream in destination storage. If the stream already
1390        * exist, it will be deleted and a new one will be created.
1391        */
1392       hr = IStorage_CreateStream( pstgDest, data.name,
1393                                   STGM_CREATE|STGM_WRITE|STGM_SHARE_EXCLUSIVE,
1394                                   0, 0, &pstrTmp );
1395 
1396       /*
1397        * open child stream storage. This operation must succeed even if the
1398        * stream is already open, so we use internal functions to do it.
1399        */
1400       if (hr == S_OK)
1401       {
1402         StgStreamImpl *streamimpl = StgStreamImpl_Construct(This, STGM_READ|STGM_SHARE_EXCLUSIVE, srcEntry);
1403 
1404         if (streamimpl)
1405         {
1406           pstrChild = &streamimpl->IStream_iface;
1407           if (pstrChild)
1408             IStream_AddRef(pstrChild);
1409         }
1410         else
1411         {
1412           pstrChild = NULL;
1413           hr = E_OUTOFMEMORY;
1414         }
1415       }
1416 
1417       if (hr == S_OK)
1418       {
1419         /*
1420          * Get the size of the source stream
1421          */
1422         IStream_Stat( pstrChild, &strStat, STATFLAG_NONAME );
1423 
1424         /*
1425          * Set the size of the destination stream.
1426          */
1427         IStream_SetSize(pstrTmp, strStat.cbSize);
1428 
1429         /*
1430          * do the copy
1431          */
1432         hr = IStream_CopyTo( pstrChild, pstrTmp, strStat.cbSize,
1433                              NULL, NULL );
1434 
1435         IStream_Release( pstrChild );
1436       }
1437 
1438       IStream_Release( pstrTmp );
1439     }
1440   }
1441 
1442   /* copy siblings */
1443   if (SUCCEEDED(hr))
1444     hr = StorageBaseImpl_CopyChildEntryTo( This, data.leftChild, skip_storage,
1445                                            skip_stream, snbExclude, pstgDest );
1446 
1447   if (SUCCEEDED(hr))
1448     hr = StorageBaseImpl_CopyChildEntryTo( This, data.rightChild, skip_storage,
1449                                            skip_stream, snbExclude, pstgDest );
1450 
1451   TRACE("<-- %08x\n", hr);
1452   return hr;
1453 }
1454 
1455 static BOOL StorageBaseImpl_IsStreamOpen(StorageBaseImpl * stg, DirRef streamEntry)
1456 {
1457   StgStreamImpl *strm;
1458 
1459   TRACE("%p,%d\n", stg, streamEntry);
1460 
1461   LIST_FOR_EACH_ENTRY(strm, &stg->strmHead, StgStreamImpl, StrmListEntry)
1462   {
1463     if (strm->dirEntry == streamEntry)
1464     {
1465       return TRUE;
1466     }
1467   }
1468 
1469   return FALSE;
1470 }
1471 
1472 static BOOL StorageBaseImpl_IsStorageOpen(StorageBaseImpl * stg, DirRef storageEntry)
1473 {
1474   StorageInternalImpl *childstg;
1475 
1476   TRACE("%p,%d\n", stg, storageEntry);
1477 
1478   LIST_FOR_EACH_ENTRY(childstg, &stg->storageHead, StorageInternalImpl, ParentListEntry)
1479   {
1480     if (childstg->base.storageDirEntry == storageEntry)
1481     {
1482       return TRUE;
1483     }
1484   }
1485 
1486   return FALSE;
1487 }
1488 
1489 /************************************************************************
1490  * StorageBaseImpl_OpenStream (IStorage)
1491  *
1492  * This method will open the specified stream object from the current storage.
1493  *
1494  * See Windows documentation for more details on IStorage methods.
1495  */
1496 static HRESULT WINAPI StorageBaseImpl_OpenStream(
1497   IStorage*        iface,
1498   const OLECHAR*   pwcsName,  /* [string][in] */
1499   void*            reserved1, /* [unique][in] */
1500   DWORD            grfMode,   /* [in]  */
1501   DWORD            reserved2, /* [in]  */
1502   IStream**        ppstm)     /* [out] */
1503 {
1504   StorageBaseImpl *This = impl_from_IStorage(iface);
1505   StgStreamImpl*    newStream;
1506   DirEntry          currentEntry;
1507   DirRef            streamEntryRef;
1508   HRESULT           res = STG_E_UNKNOWN;
1509 
1510   TRACE("(%p, %s, %p, %x, %d, %p)\n",
1511 	iface, debugstr_w(pwcsName), reserved1, grfMode, reserved2, ppstm);
1512 
1513   if ( (pwcsName==NULL) || (ppstm==0) )
1514   {
1515     res = E_INVALIDARG;
1516     goto end;
1517   }
1518 
1519   *ppstm = NULL;
1520 
1521   if ( FAILED( validateSTGM(grfMode) ) ||
1522        STGM_SHARE_MODE(grfMode) != STGM_SHARE_EXCLUSIVE)
1523   {
1524     res = STG_E_INVALIDFLAG;
1525     goto end;
1526   }
1527 
1528   /*
1529    * As documented.
1530    */
1531   if ( (grfMode & STGM_DELETEONRELEASE) || (grfMode & STGM_TRANSACTED) )
1532   {
1533     res = STG_E_INVALIDFUNCTION;
1534     goto end;
1535   }
1536 
1537   if (This->reverted)
1538   {
1539     res = STG_E_REVERTED;
1540     goto end;
1541   }
1542 
1543   /*
1544    * Check that we're compatible with the parent's storage mode, but
1545    * only if we are not in transacted mode
1546    */
1547   if(!(This->openFlags & STGM_TRANSACTED)) {
1548     if ( STGM_ACCESS_MODE( grfMode ) > STGM_ACCESS_MODE( This->openFlags ) )
1549     {
1550       res = STG_E_INVALIDFLAG;
1551       goto end;
1552     }
1553   }
1554 
1555   /*
1556    * Search for the element with the given name
1557    */
1558   streamEntryRef = findElement(
1559     This,
1560     This->storageDirEntry,
1561     pwcsName,
1562     &currentEntry);
1563 
1564   /*
1565    * If it was found, construct the stream object and return a pointer to it.
1566    */
1567   if ( (streamEntryRef!=DIRENTRY_NULL) &&
1568        (currentEntry.stgType==STGTY_STREAM) )
1569   {
1570     if (StorageBaseImpl_IsStreamOpen(This, streamEntryRef))
1571     {
1572       /* A single stream cannot be opened a second time. */
1573       res = STG_E_ACCESSDENIED;
1574       goto end;
1575     }
1576 
1577     newStream = StgStreamImpl_Construct(This, grfMode, streamEntryRef);
1578 
1579     if (newStream)
1580     {
1581       newStream->grfMode = grfMode;
1582       *ppstm = &newStream->IStream_iface;
1583 
1584       IStream_AddRef(*ppstm);
1585 
1586       res = S_OK;
1587       goto end;
1588     }
1589 
1590     res = E_OUTOFMEMORY;
1591     goto end;
1592   }
1593 
1594   res = STG_E_FILENOTFOUND;
1595 
1596 end:
1597   if (res == S_OK)
1598     TRACE("<-- IStream %p\n", *ppstm);
1599   TRACE("<-- %08x\n", res);
1600   return res;
1601 }
1602 
1603 /************************************************************************
1604  * StorageBaseImpl_OpenStorage (IStorage)
1605  *
1606  * This method will open a new storage object from the current storage.
1607  *
1608  * See Windows documentation for more details on IStorage methods.
1609  */
1610 static HRESULT WINAPI StorageBaseImpl_OpenStorage(
1611   IStorage*        iface,
1612   const OLECHAR*   pwcsName,      /* [string][unique][in] */
1613   IStorage*        pstgPriority,  /* [unique][in] */
1614   DWORD            grfMode,       /* [in] */
1615   SNB              snbExclude,    /* [unique][in] */
1616   DWORD            reserved,      /* [in] */
1617   IStorage**       ppstg)         /* [out] */
1618 {
1619   StorageBaseImpl *This = impl_from_IStorage(iface);
1620   StorageInternalImpl*   newStorage;
1621   StorageBaseImpl*       newTransactedStorage;
1622   DirEntry               currentEntry;
1623   DirRef                 storageEntryRef;
1624   HRESULT                res = STG_E_UNKNOWN;
1625 
1626   TRACE("(%p, %s, %p, %x, %p, %d, %p)\n",
1627 	iface, debugstr_w(pwcsName), pstgPriority,
1628 	grfMode, snbExclude, reserved, ppstg);
1629 
1630   if ((pwcsName==NULL) || (ppstg==0) )
1631   {
1632     res = E_INVALIDARG;
1633     goto end;
1634   }
1635 
1636   if (This->openFlags & STGM_SIMPLE)
1637   {
1638     res = STG_E_INVALIDFUNCTION;
1639     goto end;
1640   }
1641 
1642   /* as documented */
1643   if (snbExclude != NULL)
1644   {
1645     res = STG_E_INVALIDPARAMETER;
1646     goto end;
1647   }
1648 
1649   if ( FAILED( validateSTGM(grfMode) ))
1650   {
1651     res = STG_E_INVALIDFLAG;
1652     goto end;
1653   }
1654 
1655   /*
1656    * As documented.
1657    */
1658   if ( STGM_SHARE_MODE(grfMode) != STGM_SHARE_EXCLUSIVE ||
1659         (grfMode & STGM_DELETEONRELEASE) ||
1660         (grfMode & STGM_PRIORITY) )
1661   {
1662     res = STG_E_INVALIDFUNCTION;
1663     goto end;
1664   }
1665 
1666   if (This->reverted)
1667     return STG_E_REVERTED;
1668 
1669   /*
1670    * Check that we're compatible with the parent's storage mode,
1671    * but only if we are not transacted
1672    */
1673   if(!(This->openFlags & STGM_TRANSACTED)) {
1674     if ( STGM_ACCESS_MODE( grfMode ) > STGM_ACCESS_MODE( This->openFlags ) )
1675     {
1676       res = STG_E_ACCESSDENIED;
1677       goto end;
1678     }
1679   }
1680 
1681   *ppstg = NULL;
1682 
1683   storageEntryRef = findElement(
1684                          This,
1685                          This->storageDirEntry,
1686                          pwcsName,
1687                          &currentEntry);
1688 
1689   if ( (storageEntryRef!=DIRENTRY_NULL) &&
1690        (currentEntry.stgType==STGTY_STORAGE) )
1691   {
1692     if (StorageBaseImpl_IsStorageOpen(This, storageEntryRef))
1693     {
1694       /* A single storage cannot be opened a second time. */
1695       res = STG_E_ACCESSDENIED;
1696       goto end;
1697     }
1698 
1699     newStorage = StorageInternalImpl_Construct(
1700                    This,
1701                    grfMode,
1702                    storageEntryRef);
1703 
1704     if (newStorage != 0)
1705     {
1706       if (grfMode & STGM_TRANSACTED)
1707       {
1708         res = Storage_ConstructTransacted(&newStorage->base, FALSE, &newTransactedStorage);
1709 
1710         if (FAILED(res))
1711         {
1712           HeapFree(GetProcessHeap(), 0, newStorage);
1713           goto end;
1714         }
1715 
1716         *ppstg = &newTransactedStorage->IStorage_iface;
1717       }
1718       else
1719       {
1720         *ppstg = &newStorage->base.IStorage_iface;
1721       }
1722 
1723       list_add_tail(&This->storageHead, &newStorage->ParentListEntry);
1724 
1725       res = S_OK;
1726       goto end;
1727     }
1728 
1729     res = STG_E_INSUFFICIENTMEMORY;
1730     goto end;
1731   }
1732 
1733   res = STG_E_FILENOTFOUND;
1734 
1735 end:
1736   TRACE("<-- %08x\n", res);
1737   return res;
1738 }
1739 
1740 /************************************************************************
1741  * StorageBaseImpl_EnumElements (IStorage)
1742  *
1743  * This method will create an enumerator object that can be used to
1744  * retrieve information about all the elements in the storage object.
1745  *
1746  * See Windows documentation for more details on IStorage methods.
1747  */
1748 static HRESULT WINAPI StorageBaseImpl_EnumElements(
1749   IStorage*       iface,
1750   DWORD           reserved1, /* [in] */
1751   void*           reserved2, /* [size_is][unique][in] */
1752   DWORD           reserved3, /* [in] */
1753   IEnumSTATSTG**  ppenum)    /* [out] */
1754 {
1755   StorageBaseImpl *This = impl_from_IStorage(iface);
1756   IEnumSTATSTGImpl* newEnum;
1757 
1758   TRACE("(%p, %d, %p, %d, %p)\n",
1759 	iface, reserved1, reserved2, reserved3, ppenum);
1760 
1761   if (!ppenum)
1762     return E_INVALIDARG;
1763 
1764   if (This->reverted)
1765     return STG_E_REVERTED;
1766 
1767   newEnum = IEnumSTATSTGImpl_Construct(
1768               This,
1769               This->storageDirEntry);
1770 
1771   if (newEnum)
1772   {
1773     *ppenum = &newEnum->IEnumSTATSTG_iface;
1774     return S_OK;
1775   }
1776 
1777   return E_OUTOFMEMORY;
1778 }
1779 
1780 /************************************************************************
1781  * StorageBaseImpl_Stat (IStorage)
1782  *
1783  * This method will retrieve information about this storage object.
1784  *
1785  * See Windows documentation for more details on IStorage methods.
1786  */
1787 static HRESULT WINAPI StorageBaseImpl_Stat(
1788   IStorage*        iface,
1789   STATSTG*         pstatstg,     /* [out] */
1790   DWORD            grfStatFlag)  /* [in] */
1791 {
1792   StorageBaseImpl *This = impl_from_IStorage(iface);
1793   DirEntry       currentEntry;
1794   HRESULT        res = STG_E_UNKNOWN;
1795 
1796   TRACE("(%p, %p, %x)\n",
1797 	iface, pstatstg, grfStatFlag);
1798 
1799   if (!pstatstg)
1800   {
1801     res = E_INVALIDARG;
1802     goto end;
1803   }
1804 
1805   if (This->reverted)
1806   {
1807     res = STG_E_REVERTED;
1808     goto end;
1809   }
1810 
1811   res = StorageBaseImpl_ReadDirEntry(
1812                     This,
1813                     This->storageDirEntry,
1814                     &currentEntry);
1815 
1816   if (SUCCEEDED(res))
1817   {
1818     StorageUtl_CopyDirEntryToSTATSTG(
1819       This,
1820       pstatstg,
1821       &currentEntry,
1822       grfStatFlag);
1823 
1824     pstatstg->grfMode = This->openFlags;
1825     pstatstg->grfStateBits = This->stateBits;
1826   }
1827 
1828 end:
1829   if (res == S_OK)
1830   {
1831     TRACE("<-- STATSTG: pwcsName: %s, type: %d, cbSize.Low/High: %d/%d, grfMode: %08x, grfLocksSupported: %d, grfStateBits: %08x\n", debugstr_w(pstatstg->pwcsName), pstatstg->type, pstatstg->cbSize.u.LowPart, pstatstg->cbSize.u.HighPart, pstatstg->grfMode, pstatstg->grfLocksSupported, pstatstg->grfStateBits);
1832   }
1833   TRACE("<-- %08x\n", res);
1834   return res;
1835 }
1836 
1837 /************************************************************************
1838  * StorageBaseImpl_RenameElement (IStorage)
1839  *
1840  * This method will rename the specified element.
1841  *
1842  * See Windows documentation for more details on IStorage methods.
1843  */
1844 static HRESULT WINAPI StorageBaseImpl_RenameElement(
1845             IStorage*        iface,
1846             const OLECHAR*   pwcsOldName,  /* [in] */
1847             const OLECHAR*   pwcsNewName)  /* [in] */
1848 {
1849   StorageBaseImpl *This = impl_from_IStorage(iface);
1850   DirEntry          currentEntry;
1851   DirRef            currentEntryRef;
1852 
1853   TRACE("(%p, %s, %s)\n",
1854 	iface, debugstr_w(pwcsOldName), debugstr_w(pwcsNewName));
1855 
1856   if (This->reverted)
1857     return STG_E_REVERTED;
1858 
1859   currentEntryRef = findElement(This,
1860                                    This->storageDirEntry,
1861                                    pwcsNewName,
1862                                    &currentEntry);
1863 
1864   if (currentEntryRef != DIRENTRY_NULL)
1865   {
1866     /*
1867      * There is already an element with the new name
1868      */
1869     return STG_E_FILEALREADYEXISTS;
1870   }
1871 
1872   /*
1873    * Search for the old element name
1874    */
1875   currentEntryRef = findElement(This,
1876                                    This->storageDirEntry,
1877                                    pwcsOldName,
1878                                    &currentEntry);
1879 
1880   if (currentEntryRef != DIRENTRY_NULL)
1881   {
1882     if (StorageBaseImpl_IsStreamOpen(This, currentEntryRef) ||
1883         StorageBaseImpl_IsStorageOpen(This, currentEntryRef))
1884     {
1885       WARN("Element is already open; cannot rename.\n");
1886       return STG_E_ACCESSDENIED;
1887     }
1888 
1889     /* Remove the element from its current position in the tree */
1890     removeFromTree(This, This->storageDirEntry,
1891         currentEntryRef);
1892 
1893     /* Change the name of the element */
1894     strcpyW(currentEntry.name, pwcsNewName);
1895 
1896     /* Delete any sibling links */
1897     currentEntry.leftChild = DIRENTRY_NULL;
1898     currentEntry.rightChild = DIRENTRY_NULL;
1899 
1900     StorageBaseImpl_WriteDirEntry(This, currentEntryRef,
1901         &currentEntry);
1902 
1903     /* Insert the element in a new position in the tree */
1904     insertIntoTree(This, This->storageDirEntry,
1905         currentEntryRef);
1906   }
1907   else
1908   {
1909     /*
1910      * There is no element with the old name
1911      */
1912     return STG_E_FILENOTFOUND;
1913   }
1914 
1915   return StorageBaseImpl_Flush(This);
1916 }
1917 
1918 /************************************************************************
1919  * StorageBaseImpl_CreateStream (IStorage)
1920  *
1921  * This method will create a stream object within this storage
1922  *
1923  * See Windows documentation for more details on IStorage methods.
1924  */
1925 static HRESULT WINAPI StorageBaseImpl_CreateStream(
1926             IStorage*        iface,
1927             const OLECHAR*   pwcsName,  /* [string][in] */
1928             DWORD            grfMode,   /* [in] */
1929             DWORD            reserved1, /* [in] */
1930             DWORD            reserved2, /* [in] */
1931             IStream**        ppstm)     /* [out] */
1932 {
1933   StorageBaseImpl *This = impl_from_IStorage(iface);
1934   StgStreamImpl*    newStream;
1935   DirEntry          currentEntry, newStreamEntry;
1936   DirRef            currentEntryRef, newStreamEntryRef;
1937   HRESULT hr;
1938 
1939   TRACE("(%p, %s, %x, %d, %d, %p)\n",
1940 	iface, debugstr_w(pwcsName), grfMode,
1941 	reserved1, reserved2, ppstm);
1942 
1943   if (ppstm == 0)
1944     return STG_E_INVALIDPOINTER;
1945 
1946   if (pwcsName == 0)
1947     return STG_E_INVALIDNAME;
1948 
1949   if (reserved1 || reserved2)
1950     return STG_E_INVALIDPARAMETER;
1951 
1952   if ( FAILED( validateSTGM(grfMode) ))
1953     return STG_E_INVALIDFLAG;
1954 
1955   if (STGM_SHARE_MODE(grfMode) != STGM_SHARE_EXCLUSIVE)
1956     return STG_E_INVALIDFLAG;
1957 
1958   if (This->reverted)
1959     return STG_E_REVERTED;
1960 
1961   /*
1962    * As documented.
1963    */
1964   if ((grfMode & STGM_DELETEONRELEASE) ||
1965       (grfMode & STGM_TRANSACTED))
1966     return STG_E_INVALIDFUNCTION;
1967 
1968   /*
1969    * Don't worry about permissions in transacted mode, as we can always write
1970    * changes; we just can't always commit them.
1971    */
1972   if(!(This->openFlags & STGM_TRANSACTED)) {
1973     /* Can't create a stream on read-only storage */
1974     if ( STGM_ACCESS_MODE( This->openFlags ) == STGM_READ )
1975       return STG_E_ACCESSDENIED;
1976 
1977     /* Can't create a stream with greater access than the parent. */
1978     if ( STGM_ACCESS_MODE( grfMode ) > STGM_ACCESS_MODE( This->openFlags ) )
1979       return STG_E_ACCESSDENIED;
1980   }
1981 
1982   if(This->openFlags & STGM_SIMPLE)
1983     if(grfMode & STGM_CREATE) return STG_E_INVALIDFLAG;
1984 
1985   *ppstm = 0;
1986 
1987   currentEntryRef = findElement(This,
1988                                    This->storageDirEntry,
1989                                    pwcsName,
1990                                    &currentEntry);
1991 
1992   if (currentEntryRef != DIRENTRY_NULL)
1993   {
1994     /*
1995      * An element with this name already exists
1996      */
1997     if (STGM_CREATE_MODE(grfMode) == STGM_CREATE)
1998     {
1999       IStorage_DestroyElement(iface, pwcsName);
2000     }
2001     else
2002       return STG_E_FILEALREADYEXISTS;
2003   }
2004 
2005   /*
2006    * memset the empty entry
2007    */
2008   memset(&newStreamEntry, 0, sizeof(DirEntry));
2009 
2010   newStreamEntry.sizeOfNameString =
2011       ( lstrlenW(pwcsName)+1 ) * sizeof(WCHAR);
2012 
2013   if (newStreamEntry.sizeOfNameString > DIRENTRY_NAME_BUFFER_LEN)
2014     return STG_E_INVALIDNAME;
2015 
2016   strcpyW(newStreamEntry.name, pwcsName);
2017 
2018   newStreamEntry.stgType       = STGTY_STREAM;
2019   newStreamEntry.startingBlock = BLOCK_END_OF_CHAIN;
2020   newStreamEntry.size.u.LowPart  = 0;
2021   newStreamEntry.size.u.HighPart = 0;
2022 
2023   newStreamEntry.leftChild        = DIRENTRY_NULL;
2024   newStreamEntry.rightChild       = DIRENTRY_NULL;
2025   newStreamEntry.dirRootEntry     = DIRENTRY_NULL;
2026 
2027   /* call CoFileTime to get the current time
2028   newStreamEntry.ctime
2029   newStreamEntry.mtime
2030   */
2031 
2032   /*  newStreamEntry.clsid */
2033 
2034   /*
2035    * Create an entry with the new data
2036    */
2037   hr = StorageBaseImpl_CreateDirEntry(This, &newStreamEntry, &newStreamEntryRef);
2038   if (FAILED(hr))
2039     return hr;
2040 
2041   /*
2042    * Insert the new entry in the parent storage's tree.
2043    */
2044   hr = insertIntoTree(
2045     This,
2046     This->storageDirEntry,
2047     newStreamEntryRef);
2048   if (FAILED(hr))
2049   {
2050     StorageBaseImpl_DestroyDirEntry(This, newStreamEntryRef);
2051     return hr;
2052   }
2053 
2054   /*
2055    * Open the stream to return it.
2056    */
2057   newStream = StgStreamImpl_Construct(This, grfMode, newStreamEntryRef);
2058 
2059   if (newStream)
2060   {
2061     *ppstm = &newStream->IStream_iface;
2062     IStream_AddRef(*ppstm);
2063   }
2064   else
2065   {
2066     return STG_E_INSUFFICIENTMEMORY;
2067   }
2068 
2069   return StorageBaseImpl_Flush(This);
2070 }
2071 
2072 /************************************************************************
2073  * StorageBaseImpl_SetClass (IStorage)
2074  *
2075  * This method will write the specified CLSID in the directory entry of this
2076  * storage.
2077  *
2078  * See Windows documentation for more details on IStorage methods.
2079  */
2080 static HRESULT WINAPI StorageBaseImpl_SetClass(
2081   IStorage*        iface,
2082   REFCLSID         clsid) /* [in] */
2083 {
2084   StorageBaseImpl *This = impl_from_IStorage(iface);
2085   HRESULT hRes;
2086   DirEntry currentEntry;
2087 
2088   TRACE("(%p, %s)\n", iface, wine_dbgstr_guid(clsid));
2089 
2090   if (This->reverted)
2091     return STG_E_REVERTED;
2092 
2093   hRes = StorageBaseImpl_ReadDirEntry(This,
2094                                       This->storageDirEntry,
2095                                       &currentEntry);
2096   if (SUCCEEDED(hRes))
2097   {
2098     currentEntry.clsid = *clsid;
2099 
2100     hRes = StorageBaseImpl_WriteDirEntry(This,
2101                                          This->storageDirEntry,
2102                                          &currentEntry);
2103   }
2104 
2105   if (SUCCEEDED(hRes))
2106     hRes = StorageBaseImpl_Flush(This);
2107 
2108   return hRes;
2109 }
2110 
2111 /************************************************************************
2112  * StorageBaseImpl_CreateStorage (IStorage)
2113  *
2114  * This method will create the storage object within the provided storage.
2115  *
2116  * See Windows documentation for more details on IStorage methods.
2117  */
2118 static HRESULT WINAPI StorageBaseImpl_CreateStorage(
2119   IStorage*      iface,
2120   const OLECHAR  *pwcsName, /* [string][in] */
2121   DWORD            grfMode,   /* [in] */
2122   DWORD            reserved1, /* [in] */
2123   DWORD            reserved2, /* [in] */
2124   IStorage       **ppstg)   /* [out] */
2125 {
2126   StorageBaseImpl* This = impl_from_IStorage(iface);
2127 
2128   DirEntry         currentEntry;
2129   DirEntry         newEntry;
2130   DirRef           currentEntryRef;
2131   DirRef           newEntryRef;
2132   HRESULT          hr;
2133 
2134   TRACE("(%p, %s, %x, %d, %d, %p)\n",
2135 	iface, debugstr_w(pwcsName), grfMode,
2136 	reserved1, reserved2, ppstg);
2137 
2138   if (ppstg == 0)
2139     return STG_E_INVALIDPOINTER;
2140 
2141   if (This->openFlags & STGM_SIMPLE)
2142   {
2143     return STG_E_INVALIDFUNCTION;
2144   }
2145 
2146   if (pwcsName == 0)
2147     return STG_E_INVALIDNAME;
2148 
2149   *ppstg = NULL;
2150 
2151   if ( FAILED( validateSTGM(grfMode) ) ||
2152        (grfMode & STGM_DELETEONRELEASE) )
2153   {
2154     WARN("bad grfMode: 0x%x\n", grfMode);
2155     return STG_E_INVALIDFLAG;
2156   }
2157 
2158   if (This->reverted)
2159     return STG_E_REVERTED;
2160 
2161   /*
2162    * Check that we're compatible with the parent's storage mode
2163    */
2164   if ( !(This->openFlags & STGM_TRANSACTED) &&
2165        STGM_ACCESS_MODE( grfMode ) > STGM_ACCESS_MODE( This->openFlags ) )
2166   {
2167     WARN("access denied\n");
2168     return STG_E_ACCESSDENIED;
2169   }
2170 
2171   currentEntryRef = findElement(This,
2172                                    This->storageDirEntry,
2173                                    pwcsName,
2174                                    &currentEntry);
2175 
2176   if (currentEntryRef != DIRENTRY_NULL)
2177   {
2178     /*
2179      * An element with this name already exists
2180      */
2181     if (STGM_CREATE_MODE(grfMode) == STGM_CREATE &&
2182         ((This->openFlags & STGM_TRANSACTED) ||
2183          STGM_ACCESS_MODE(This->openFlags) != STGM_READ))
2184     {
2185       hr = IStorage_DestroyElement(iface, pwcsName);
2186       if (FAILED(hr))
2187         return hr;
2188     }
2189     else
2190     {
2191       WARN("file already exists\n");
2192       return STG_E_FILEALREADYEXISTS;
2193     }
2194   }
2195   else if (!(This->openFlags & STGM_TRANSACTED) &&
2196            STGM_ACCESS_MODE(This->openFlags) == STGM_READ)
2197   {
2198     WARN("read-only storage\n");
2199     return STG_E_ACCESSDENIED;
2200   }
2201 
2202   memset(&newEntry, 0, sizeof(DirEntry));
2203 
2204   newEntry.sizeOfNameString = (lstrlenW(pwcsName)+1)*sizeof(WCHAR);
2205 
2206   if (newEntry.sizeOfNameString > DIRENTRY_NAME_BUFFER_LEN)
2207   {
2208     FIXME("name too long\n");
2209     return STG_E_INVALIDNAME;
2210   }
2211 
2212   strcpyW(newEntry.name, pwcsName);
2213 
2214   newEntry.stgType       = STGTY_STORAGE;
2215   newEntry.startingBlock = BLOCK_END_OF_CHAIN;
2216   newEntry.size.u.LowPart  = 0;
2217   newEntry.size.u.HighPart = 0;
2218 
2219   newEntry.leftChild        = DIRENTRY_NULL;
2220   newEntry.rightChild       = DIRENTRY_NULL;
2221   newEntry.dirRootEntry     = DIRENTRY_NULL;
2222 
2223   /* call CoFileTime to get the current time
2224   newEntry.ctime
2225   newEntry.mtime
2226   */
2227 
2228   /*  newEntry.clsid */
2229 
2230   /*
2231    * Create a new directory entry for the storage
2232    */
2233   hr = StorageBaseImpl_CreateDirEntry(This, &newEntry, &newEntryRef);
2234   if (FAILED(hr))
2235     return hr;
2236 
2237   /*
2238    * Insert the new directory entry into the parent storage's tree
2239    */
2240   hr = insertIntoTree(
2241     This,
2242     This->storageDirEntry,
2243     newEntryRef);
2244   if (FAILED(hr))
2245   {
2246     StorageBaseImpl_DestroyDirEntry(This, newEntryRef);
2247     return hr;
2248   }
2249 
2250   /*
2251    * Open it to get a pointer to return.
2252    */
2253   hr = IStorage_OpenStorage(iface, pwcsName, 0, grfMode, 0, 0, ppstg);
2254 
2255   if( (hr != S_OK) || (*ppstg == NULL))
2256   {
2257     return hr;
2258   }
2259 
2260   if (SUCCEEDED(hr))
2261     hr = StorageBaseImpl_Flush(This);
2262 
2263   return S_OK;
2264 }
2265 
2266 static HRESULT StorageBaseImpl_CopyStorageEntryTo(StorageBaseImpl *This,
2267     DirRef srcEntry, BOOL skip_storage, BOOL skip_stream,
2268     SNB snbExclude, IStorage *pstgDest)
2269 {
2270   DirEntry data;
2271   HRESULT hr;
2272 
2273   hr = StorageBaseImpl_ReadDirEntry( This, srcEntry, &data );
2274 
2275   if (SUCCEEDED(hr))
2276     hr = IStorage_SetClass( pstgDest, &data.clsid );
2277 
2278   if (SUCCEEDED(hr))
2279     hr = StorageBaseImpl_CopyChildEntryTo( This, data.dirRootEntry, skip_storage,
2280       skip_stream, snbExclude, pstgDest );
2281 
2282   TRACE("<-- %08x\n", hr);
2283   return hr;
2284 }
2285 
2286 /*************************************************************************
2287  * CopyTo (IStorage)
2288  */
2289 static HRESULT WINAPI StorageBaseImpl_CopyTo(
2290   IStorage*   iface,
2291   DWORD       ciidExclude,  /* [in] */
2292   const IID*  rgiidExclude, /* [size_is][unique][in] */
2293   SNB         snbExclude,   /* [unique][in] */
2294   IStorage*   pstgDest)     /* [unique][in] */
2295 {
2296   StorageBaseImpl *This = impl_from_IStorage(iface);
2297 
2298   BOOL         skip_storage = FALSE, skip_stream = FALSE;
2299   DWORD        i;
2300 
2301   TRACE("(%p, %d, %p, %p, %p)\n",
2302 	iface, ciidExclude, rgiidExclude,
2303 	snbExclude, pstgDest);
2304 
2305   if ( pstgDest == 0 )
2306     return STG_E_INVALIDPOINTER;
2307 
2308   for(i = 0; i < ciidExclude; ++i)
2309   {
2310     if(IsEqualGUID(&IID_IStorage, &rgiidExclude[i]))
2311         skip_storage = TRUE;
2312     else if(IsEqualGUID(&IID_IStream, &rgiidExclude[i]))
2313         skip_stream = TRUE;
2314     else
2315         WARN("Unknown excluded GUID: %s\n", debugstr_guid(&rgiidExclude[i]));
2316   }
2317 
2318   if (!skip_storage)
2319   {
2320     /* Give up early if it looks like this would be infinitely recursive.
2321      * Oddly enough, this includes some cases that aren't really recursive, like
2322      * copying to a transacted child. */
2323     IStorage *pstgDestAncestor = pstgDest;
2324     IStorage *pstgDestAncestorChild = NULL;
2325 
2326     /* Go up the chain from the destination until we find the source storage. */
2327     while (pstgDestAncestor != iface) {
2328       pstgDestAncestorChild = pstgDest;
2329 
2330       if (pstgDestAncestor->lpVtbl == &TransactedSnapshotImpl_Vtbl)
2331       {
2332         TransactedSnapshotImpl *snapshot = (TransactedSnapshotImpl*) pstgDestAncestor;
2333 
2334         pstgDestAncestor = &snapshot->transactedParent->IStorage_iface;
2335       }
2336       else if (pstgDestAncestor->lpVtbl == &StorageInternalImpl_Vtbl)
2337       {
2338         StorageInternalImpl *internal = (StorageInternalImpl*) pstgDestAncestor;
2339 
2340         pstgDestAncestor = &internal->parentStorage->IStorage_iface;
2341       }
2342       else
2343         break;
2344     }
2345 
2346     if (pstgDestAncestor == iface)
2347     {
2348       BOOL fail = TRUE;
2349 
2350       if (pstgDestAncestorChild && snbExclude)
2351       {
2352         StorageBaseImpl *ancestorChildBase = (StorageBaseImpl*)pstgDestAncestorChild;
2353         DirEntry data;
2354         WCHAR **snb = snbExclude;
2355 
2356         StorageBaseImpl_ReadDirEntry(ancestorChildBase, ancestorChildBase->storageDirEntry, &data);
2357 
2358         while ( *snb != NULL && fail )
2359         {
2360           if ( lstrcmpW(data.name, *snb) == 0 )
2361             fail = FALSE;
2362           ++snb;
2363         }
2364       }
2365 
2366       if (fail)
2367         return STG_E_ACCESSDENIED;
2368     }
2369   }
2370 
2371   return StorageBaseImpl_CopyStorageEntryTo( This, This->storageDirEntry,
2372     skip_storage, skip_stream, snbExclude, pstgDest );
2373 }
2374 
2375 /*************************************************************************
2376  * MoveElementTo (IStorage)
2377  */
2378 static HRESULT WINAPI StorageBaseImpl_MoveElementTo(
2379   IStorage*     iface,
2380   const OLECHAR *pwcsName,   /* [string][in] */
2381   IStorage      *pstgDest,   /* [unique][in] */
2382   const OLECHAR *pwcsNewName,/* [string][in] */
2383   DWORD           grfFlags)    /* [in] */
2384 {
2385   FIXME("(%p %s %p %s %u): stub\n", iface,
2386          debugstr_w(pwcsName), pstgDest,
2387          debugstr_w(pwcsNewName), grfFlags);
2388   return E_NOTIMPL;
2389 }
2390 
2391 /*************************************************************************
2392  * Commit (IStorage)
2393  *
2394  * Ensures that any changes made to a storage object open in transacted mode
2395  * are reflected in the parent storage
2396  *
2397  * In a non-transacted mode, this ensures all cached writes are completed.
2398  */
2399 static HRESULT WINAPI StorageBaseImpl_Commit(
2400   IStorage*   iface,
2401   DWORD         grfCommitFlags)/* [in] */
2402 {
2403   StorageBaseImpl* This = impl_from_IStorage(iface);
2404   TRACE("(%p %d)\n", iface, grfCommitFlags);
2405   return StorageBaseImpl_Flush(This);
2406 }
2407 
2408 /*************************************************************************
2409  * Revert (IStorage)
2410  *
2411  * Discard all changes that have been made since the last commit operation
2412  */
2413 static HRESULT WINAPI StorageBaseImpl_Revert(
2414   IStorage* iface)
2415 {
2416   TRACE("(%p)\n", iface);
2417   return S_OK;
2418 }
2419 
2420 /*********************************************************************
2421  *
2422  * Internal helper function for StorageBaseImpl_DestroyElement()
2423  *
2424  * Delete the contents of a storage entry.
2425  *
2426  */
2427 static HRESULT deleteStorageContents(
2428   StorageBaseImpl *parentStorage,
2429   DirRef       indexToDelete,
2430   DirEntry     entryDataToDelete)
2431 {
2432   IEnumSTATSTG *elements     = 0;
2433   IStorage   *childStorage = 0;
2434   STATSTG      currentElement;
2435   HRESULT      hr;
2436   HRESULT      destroyHr = S_OK;
2437   StorageInternalImpl *stg, *stg2;
2438 
2439   TRACE("%p,%d\n", parentStorage, indexToDelete);
2440 
2441   /* Invalidate any open storage objects. */
2442   LIST_FOR_EACH_ENTRY_SAFE(stg, stg2, &parentStorage->storageHead, StorageInternalImpl, ParentListEntry)
2443   {
2444     if (stg->base.storageDirEntry == indexToDelete)
2445     {
2446       StorageBaseImpl_Invalidate(&stg->base);
2447     }
2448   }
2449 
2450   /*
2451    * Open the storage and enumerate it
2452    */
2453   hr = IStorage_OpenStorage(
2454         &parentStorage->IStorage_iface,
2455         entryDataToDelete.name,
2456         0,
2457         STGM_WRITE | STGM_SHARE_EXCLUSIVE,
2458         0,
2459         0,
2460         &childStorage);
2461 
2462   if (hr != S_OK)
2463   {
2464     TRACE("<-- %08x\n", hr);
2465     return hr;
2466   }
2467 
2468   /*
2469    * Enumerate the elements
2470    */
2471   hr = IStorage_EnumElements(childStorage, 0, 0, 0, &elements);
2472   if (FAILED(hr))
2473   {
2474     IStorage_Release(childStorage);
2475     TRACE("<-- %08x\n", hr);
2476     return hr;
2477   }
2478 
2479   do
2480   {
2481     /*
2482      * Obtain the next element
2483      */
2484     hr = IEnumSTATSTG_Next(elements, 1, &currentElement, NULL);
2485     if (hr==S_OK)
2486     {
2487       destroyHr = IStorage_DestroyElement(childStorage, currentElement.pwcsName);
2488 
2489       CoTaskMemFree(currentElement.pwcsName);
2490     }
2491 
2492     /*
2493      * We need to Reset the enumeration every time because we delete elements
2494      * and the enumeration could be invalid
2495      */
2496     IEnumSTATSTG_Reset(elements);
2497 
2498   } while ((hr == S_OK) && (destroyHr == S_OK));
2499 
2500   IStorage_Release(childStorage);
2501   IEnumSTATSTG_Release(elements);
2502 
2503   TRACE("%08x\n", hr);
2504   return destroyHr;
2505 }
2506 
2507 /*********************************************************************
2508  *
2509  * Internal helper function for StorageBaseImpl_DestroyElement()
2510  *
2511  * Perform the deletion of a stream's data
2512  *
2513  */
2514 static HRESULT deleteStreamContents(
2515   StorageBaseImpl *parentStorage,
2516   DirRef        indexToDelete,
2517   DirEntry      entryDataToDelete)
2518 {
2519   IStream      *pis;
2520   HRESULT        hr;
2521   ULARGE_INTEGER size;
2522   StgStreamImpl *strm, *strm2;
2523 
2524   /* Invalidate any open stream objects. */
2525   LIST_FOR_EACH_ENTRY_SAFE(strm, strm2, &parentStorage->strmHead, StgStreamImpl, StrmListEntry)
2526   {
2527     if (strm->dirEntry == indexToDelete)
2528     {
2529       TRACE("Stream deleted %p\n", strm);
2530       strm->parentStorage = NULL;
2531       list_remove(&strm->StrmListEntry);
2532     }
2533   }
2534 
2535   size.u.HighPart = 0;
2536   size.u.LowPart = 0;
2537 
2538   hr = IStorage_OpenStream(&parentStorage->IStorage_iface,
2539         entryDataToDelete.name, NULL, STGM_WRITE | STGM_SHARE_EXCLUSIVE, 0, &pis);
2540 
2541   if (hr!=S_OK)
2542   {
2543     TRACE("<-- %08x\n", hr);
2544     return(hr);
2545   }
2546 
2547   /*
2548    * Zap the stream
2549    */
2550   hr = IStream_SetSize(pis, size);
2551 
2552   if(hr != S_OK)
2553   {
2554     TRACE("<-- %08x\n", hr);
2555     return hr;
2556   }
2557 
2558   /*
2559    * Release the stream object.
2560    */
2561   IStream_Release(pis);
2562   TRACE("<-- %08x\n", hr);
2563   return S_OK;
2564 }
2565 
2566 /*************************************************************************
2567  * DestroyElement (IStorage)
2568  *
2569  * Strategy: This implementation is built this way for simplicity not for speed.
2570  *          I always delete the topmost element of the enumeration and adjust
2571  *          the deleted element pointer all the time.  This takes longer to
2572  *          do but allows reinvoking DestroyElement whenever we encounter a
2573  *          storage object.  The optimisation resides in the usage of another
2574  *          enumeration strategy that would give all the leaves of a storage
2575  *          first. (postfix order)
2576  */
2577 static HRESULT WINAPI StorageBaseImpl_DestroyElement(
2578   IStorage*     iface,
2579   const OLECHAR *pwcsName)/* [string][in] */
2580 {
2581   StorageBaseImpl *This = impl_from_IStorage(iface);
2582 
2583   HRESULT           hr = S_OK;
2584   DirEntry          entryToDelete;
2585   DirRef            entryToDeleteRef;
2586 
2587   TRACE("(%p, %s)\n",
2588 	iface, debugstr_w(pwcsName));
2589 
2590   if (pwcsName==NULL)
2591     return STG_E_INVALIDPOINTER;
2592 
2593   if (This->reverted)
2594     return STG_E_REVERTED;
2595 
2596   if ( !(This->openFlags & STGM_TRANSACTED) &&
2597        STGM_ACCESS_MODE( This->openFlags ) == STGM_READ )
2598     return STG_E_ACCESSDENIED;
2599 
2600   entryToDeleteRef = findElement(
2601     This,
2602     This->storageDirEntry,
2603     pwcsName,
2604     &entryToDelete);
2605 
2606   if ( entryToDeleteRef == DIRENTRY_NULL )
2607   {
2608     TRACE("<-- STG_E_FILENOTFOUND\n");
2609     return STG_E_FILENOTFOUND;
2610   }
2611 
2612   if ( entryToDelete.stgType == STGTY_STORAGE )
2613   {
2614     hr = deleteStorageContents(
2615            This,
2616            entryToDeleteRef,
2617            entryToDelete);
2618   }
2619   else if ( entryToDelete.stgType == STGTY_STREAM )
2620   {
2621     hr = deleteStreamContents(
2622            This,
2623            entryToDeleteRef,
2624            entryToDelete);
2625   }
2626 
2627   if (hr!=S_OK)
2628   {
2629     TRACE("<-- %08x\n", hr);
2630     return hr;
2631   }
2632 
2633   /*
2634    * Remove the entry from its parent storage
2635    */
2636   hr = removeFromTree(
2637         This,
2638         This->storageDirEntry,
2639         entryToDeleteRef);
2640 
2641   /*
2642    * Invalidate the entry
2643    */
2644   if (SUCCEEDED(hr))
2645     StorageBaseImpl_DestroyDirEntry(This, entryToDeleteRef);
2646 
2647   if (SUCCEEDED(hr))
2648     hr = StorageBaseImpl_Flush(This);
2649 
2650   TRACE("<-- %08x\n", hr);
2651   return hr;
2652 }
2653 
2654 static void StorageBaseImpl_DeleteAll(StorageBaseImpl * stg)
2655 {
2656   struct list *cur, *cur2;
2657   StgStreamImpl *strm=NULL;
2658   StorageInternalImpl *childstg=NULL;
2659 
2660   LIST_FOR_EACH_SAFE(cur, cur2, &stg->strmHead) {
2661     strm = LIST_ENTRY(cur,StgStreamImpl,StrmListEntry);
2662     TRACE("Streams invalidated (stg=%p strm=%p next=%p prev=%p)\n", stg,strm,cur->next,cur->prev);
2663     strm->parentStorage = NULL;
2664     list_remove(cur);
2665   }
2666 
2667   LIST_FOR_EACH_SAFE(cur, cur2, &stg->storageHead) {
2668     childstg = LIST_ENTRY(cur,StorageInternalImpl,ParentListEntry);
2669     StorageBaseImpl_Invalidate( &childstg->base );
2670   }
2671 
2672   if (stg->transactedChild)
2673   {
2674     StorageBaseImpl_Invalidate(stg->transactedChild);
2675 
2676     stg->transactedChild = NULL;
2677   }
2678 }
2679 
2680 /******************************************************************************
2681  * SetElementTimes (IStorage)
2682  */
2683 static HRESULT WINAPI StorageBaseImpl_SetElementTimes(
2684   IStorage*     iface,
2685   const OLECHAR *pwcsName,/* [string][in] */
2686   const FILETIME  *pctime,  /* [in] */
2687   const FILETIME  *patime,  /* [in] */
2688   const FILETIME  *pmtime)  /* [in] */
2689 {
2690   FIXME("(%s,...), stub!\n",debugstr_w(pwcsName));
2691   return S_OK;
2692 }
2693 
2694 /******************************************************************************
2695  * SetStateBits (IStorage)
2696  */
2697 static HRESULT WINAPI StorageBaseImpl_SetStateBits(
2698   IStorage*   iface,
2699   DWORD         grfStateBits,/* [in] */
2700   DWORD         grfMask)     /* [in] */
2701 {
2702   StorageBaseImpl *This = impl_from_IStorage(iface);
2703 
2704   if (This->reverted)
2705     return STG_E_REVERTED;
2706 
2707   This->stateBits = (This->stateBits & ~grfMask) | (grfStateBits & grfMask);
2708   return S_OK;
2709 }
2710 
2711 /******************************************************************************
2712  * Internal stream list handlers
2713  */
2714 
2715 void StorageBaseImpl_AddStream(StorageBaseImpl * stg, StgStreamImpl * strm)
2716 {
2717   TRACE("Stream added (stg=%p strm=%p)\n", stg, strm);
2718   list_add_tail(&stg->strmHead,&strm->StrmListEntry);
2719 }
2720 
2721 void StorageBaseImpl_RemoveStream(StorageBaseImpl * stg, StgStreamImpl * strm)
2722 {
2723   TRACE("Stream removed (stg=%p strm=%p)\n", stg,strm);
2724   list_remove(&(strm->StrmListEntry));
2725 }
2726 
2727 static HRESULT StorageBaseImpl_CopyStream(
2728   StorageBaseImpl *dst, DirRef dst_entry,
2729   StorageBaseImpl *src, DirRef src_entry)
2730 {
2731   HRESULT hr;
2732   BYTE data[4096];
2733   DirEntry srcdata;
2734   ULARGE_INTEGER bytes_copied;
2735   ULONG bytestocopy, bytesread, byteswritten;
2736 
2737   hr = StorageBaseImpl_ReadDirEntry(src, src_entry, &srcdata);
2738 
2739   if (SUCCEEDED(hr))
2740   {
2741     hr = StorageBaseImpl_StreamSetSize(dst, dst_entry, srcdata.size);
2742 
2743     bytes_copied.QuadPart = 0;
2744     while (bytes_copied.QuadPart < srcdata.size.QuadPart && SUCCEEDED(hr))
2745     {
2746       bytestocopy = min(4096, srcdata.size.QuadPart - bytes_copied.QuadPart);
2747 
2748       hr = StorageBaseImpl_StreamReadAt(src, src_entry, bytes_copied, bytestocopy,
2749         data, &bytesread);
2750       if (SUCCEEDED(hr) && bytesread != bytestocopy) hr = STG_E_READFAULT;
2751 
2752       if (SUCCEEDED(hr))
2753         hr = StorageBaseImpl_StreamWriteAt(dst, dst_entry, bytes_copied, bytestocopy,
2754           data, &byteswritten);
2755       if (SUCCEEDED(hr))
2756       {
2757         if (byteswritten != bytestocopy) hr = STG_E_WRITEFAULT;
2758         bytes_copied.QuadPart += byteswritten;
2759       }
2760     }
2761   }
2762 
2763   return hr;
2764 }
2765 
2766 static HRESULT StorageBaseImpl_DupStorageTree(
2767   StorageBaseImpl *dst, DirRef *dst_entry,
2768   StorageBaseImpl *src, DirRef src_entry)
2769 {
2770   HRESULT hr;
2771   DirEntry data;
2772   BOOL has_stream=FALSE;
2773 
2774   if (src_entry == DIRENTRY_NULL)
2775   {
2776     *dst_entry = DIRENTRY_NULL;
2777     return S_OK;
2778   }
2779 
2780   hr = StorageBaseImpl_ReadDirEntry(src, src_entry, &data);
2781   if (SUCCEEDED(hr))
2782   {
2783     has_stream = (data.stgType == STGTY_STREAM && data.size.QuadPart != 0);
2784     data.startingBlock = BLOCK_END_OF_CHAIN;
2785     data.size.QuadPart = 0;
2786 
2787     hr = StorageBaseImpl_DupStorageTree(dst, &data.leftChild, src, data.leftChild);
2788   }
2789 
2790   if (SUCCEEDED(hr))
2791     hr = StorageBaseImpl_DupStorageTree(dst, &data.rightChild, src, data.rightChild);
2792 
2793   if (SUCCEEDED(hr))
2794     hr = StorageBaseImpl_DupStorageTree(dst, &data.dirRootEntry, src, data.dirRootEntry);
2795 
2796   if (SUCCEEDED(hr))
2797     hr = StorageBaseImpl_CreateDirEntry(dst, &data, dst_entry);
2798 
2799   if (SUCCEEDED(hr) && has_stream)
2800     hr = StorageBaseImpl_CopyStream(dst, *dst_entry, src, src_entry);
2801 
2802   return hr;
2803 }
2804 
2805 static HRESULT StorageBaseImpl_CopyStorageTree(
2806   StorageBaseImpl *dst, DirRef dst_entry,
2807   StorageBaseImpl *src, DirRef src_entry)
2808 {
2809   HRESULT hr;
2810   DirEntry src_data, dst_data;
2811   DirRef new_root_entry;
2812 
2813   hr = StorageBaseImpl_ReadDirEntry(src, src_entry, &src_data);
2814 
2815   if (SUCCEEDED(hr))
2816   {
2817     hr = StorageBaseImpl_DupStorageTree(dst, &new_root_entry, src, src_data.dirRootEntry);
2818   }
2819 
2820   if (SUCCEEDED(hr))
2821   {
2822     hr = StorageBaseImpl_ReadDirEntry(dst, dst_entry, &dst_data);
2823     dst_data.clsid = src_data.clsid;
2824     dst_data.ctime = src_data.ctime;
2825     dst_data.mtime = src_data.mtime;
2826     dst_data.dirRootEntry = new_root_entry;
2827   }
2828 
2829   if (SUCCEEDED(hr))
2830     hr = StorageBaseImpl_WriteDirEntry(dst, dst_entry, &dst_data);
2831 
2832   return hr;
2833 }
2834 
2835 static HRESULT StorageBaseImpl_DeleteStorageTree(StorageBaseImpl *This, DirRef entry, BOOL include_siblings)
2836 {
2837   HRESULT hr;
2838   DirEntry data;
2839   ULARGE_INTEGER zero;
2840 
2841   if (entry == DIRENTRY_NULL)
2842     return S_OK;
2843 
2844   zero.QuadPart = 0;
2845 
2846   hr = StorageBaseImpl_ReadDirEntry(This, entry, &data);
2847 
2848   if (SUCCEEDED(hr) && include_siblings)
2849     hr = StorageBaseImpl_DeleteStorageTree(This, data.leftChild, TRUE);
2850 
2851   if (SUCCEEDED(hr) && include_siblings)
2852     hr = StorageBaseImpl_DeleteStorageTree(This, data.rightChild, TRUE);
2853 
2854   if (SUCCEEDED(hr))
2855     hr = StorageBaseImpl_DeleteStorageTree(This, data.dirRootEntry, TRUE);
2856 
2857   if (SUCCEEDED(hr) && data.stgType == STGTY_STREAM)
2858     hr = StorageBaseImpl_StreamSetSize(This, entry, zero);
2859 
2860   if (SUCCEEDED(hr))
2861     hr = StorageBaseImpl_DestroyDirEntry(This, entry);
2862 
2863   return hr;
2864 }
2865 
2866 
2867 /************************************************************************
2868  * StorageImpl implementation
2869  ***********************************************************************/
2870 
2871 static HRESULT StorageImpl_ReadAt(StorageImpl* This,
2872   ULARGE_INTEGER offset,
2873   void*          buffer,
2874   ULONG          size,
2875   ULONG*         bytesRead)
2876 {
2877     return ILockBytes_ReadAt(This->lockBytes,offset,buffer,size,bytesRead);
2878 }
2879 
2880 static HRESULT StorageImpl_WriteAt(StorageImpl* This,
2881   ULARGE_INTEGER offset,
2882   const void*    buffer,
2883   const ULONG    size,
2884   ULONG*         bytesWritten)
2885 {
2886     return ILockBytes_WriteAt(This->lockBytes,offset,buffer,size,bytesWritten);
2887 }
2888 
2889 /******************************************************************************
2890  *      StorageImpl_LoadFileHeader
2891  *
2892  * This method will read in the file header
2893  */
2894 static HRESULT StorageImpl_LoadFileHeader(
2895           StorageImpl* This)
2896 {
2897   HRESULT hr;
2898   BYTE    headerBigBlock[HEADER_SIZE];
2899   int     index;
2900   ULARGE_INTEGER offset;
2901   DWORD bytes_read;
2902 
2903   TRACE("\n");
2904   /*
2905    * Get a pointer to the big block of data containing the header.
2906    */
2907   offset.u.HighPart = 0;
2908   offset.u.LowPart = 0;
2909   hr = StorageImpl_ReadAt(This, offset, headerBigBlock, HEADER_SIZE, &bytes_read);
2910   if (SUCCEEDED(hr) && bytes_read != HEADER_SIZE)
2911     hr = STG_E_FILENOTFOUND;
2912 
2913   /*
2914    * Extract the information from the header.
2915    */
2916   if (SUCCEEDED(hr))
2917   {
2918     /*
2919      * Check for the "magic number" signature and return an error if it is not
2920      * found.
2921      */
2922     if (memcmp(headerBigBlock, STORAGE_oldmagic, sizeof(STORAGE_oldmagic))==0)
2923     {
2924       return STG_E_OLDFORMAT;
2925     }
2926 
2927     if (memcmp(headerBigBlock, STORAGE_magic, sizeof(STORAGE_magic))!=0)
2928     {
2929       return STG_E_INVALIDHEADER;
2930     }
2931 
2932     StorageUtl_ReadWord(
2933       headerBigBlock,
2934       OFFSET_BIGBLOCKSIZEBITS,
2935       &This->bigBlockSizeBits);
2936 
2937     StorageUtl_ReadWord(
2938       headerBigBlock,
2939       OFFSET_SMALLBLOCKSIZEBITS,
2940       &This->smallBlockSizeBits);
2941 
2942     StorageUtl_ReadDWord(
2943       headerBigBlock,
2944       OFFSET_BBDEPOTCOUNT,
2945       &This->bigBlockDepotCount);
2946 
2947     StorageUtl_ReadDWord(
2948       headerBigBlock,
2949       OFFSET_ROOTSTARTBLOCK,
2950       &This->rootStartBlock);
2951 
2952     StorageUtl_ReadDWord(
2953       headerBigBlock,
2954       OFFSET_TRANSACTIONSIG,
2955       &This->transactionSig);
2956 
2957     StorageUtl_ReadDWord(
2958       headerBigBlock,
2959       OFFSET_SMALLBLOCKLIMIT,
2960       &This->smallBlockLimit);
2961 
2962     StorageUtl_ReadDWord(
2963       headerBigBlock,
2964       OFFSET_SBDEPOTSTART,
2965       &This->smallBlockDepotStart);
2966 
2967     StorageUtl_ReadDWord(
2968       headerBigBlock,
2969       OFFSET_EXTBBDEPOTSTART,
2970       &This->extBigBlockDepotStart);
2971 
2972     StorageUtl_ReadDWord(
2973       headerBigBlock,
2974       OFFSET_EXTBBDEPOTCOUNT,
2975       &This->extBigBlockDepotCount);
2976 
2977     for (index = 0; index < COUNT_BBDEPOTINHEADER; index ++)
2978     {
2979       StorageUtl_ReadDWord(
2980         headerBigBlock,
2981         OFFSET_BBDEPOTSTART + (sizeof(ULONG)*index),
2982         &(This->bigBlockDepotStart[index]));
2983     }
2984 
2985     /*
2986      * Make the bitwise arithmetic to get the size of the blocks in bytes.
2987      */
2988     This->bigBlockSize   = 0x000000001 << (DWORD)This->bigBlockSizeBits;
2989     This->smallBlockSize = 0x000000001 << (DWORD)This->smallBlockSizeBits;
2990 
2991     /*
2992      * Right now, the code is making some assumptions about the size of the
2993      * blocks, just make sure they are what we're expecting.
2994      */
2995     if ((This->bigBlockSize != MIN_BIG_BLOCK_SIZE && This->bigBlockSize != MAX_BIG_BLOCK_SIZE) ||
2996 	This->smallBlockSize != DEF_SMALL_BLOCK_SIZE ||
2997 	This->smallBlockLimit != LIMIT_TO_USE_SMALL_BLOCK)
2998     {
2999 	FIXME("Broken OLE storage file? bigblock=0x%x, smallblock=0x%x, sblimit=0x%x\n",
3000 	    This->bigBlockSize, This->smallBlockSize, This->smallBlockLimit);
3001 	hr = STG_E_INVALIDHEADER;
3002     }
3003     else
3004 	hr = S_OK;
3005   }
3006 
3007   return hr;
3008 }
3009 
3010 /******************************************************************************
3011  *      StorageImpl_SaveFileHeader
3012  *
3013  * This method will save to the file the header
3014  */
3015 static void StorageImpl_SaveFileHeader(
3016           StorageImpl* This)
3017 {
3018   BYTE   headerBigBlock[HEADER_SIZE];
3019   int    index;
3020   HRESULT hr;
3021   ULARGE_INTEGER offset;
3022   DWORD bytes_read, bytes_written;
3023   DWORD major_version, dirsectorcount;
3024 
3025   /*
3026    * Get a pointer to the big block of data containing the header.
3027    */
3028   offset.u.HighPart = 0;
3029   offset.u.LowPart = 0;
3030   hr = StorageImpl_ReadAt(This, offset, headerBigBlock, HEADER_SIZE, &bytes_read);
3031   if (SUCCEEDED(hr) && bytes_read != HEADER_SIZE)
3032     hr = STG_E_FILENOTFOUND;
3033 
3034   if (This->bigBlockSizeBits == 0x9)
3035     major_version = 3;
3036   else if (This->bigBlockSizeBits == 0xc)
3037     major_version = 4;
3038   else
3039   {
3040     ERR("invalid big block shift 0x%x\n", This->bigBlockSizeBits);
3041     major_version = 4;
3042   }
3043 
3044   /*
3045    * If the block read failed, the file is probably new.
3046    */
3047   if (FAILED(hr))
3048   {
3049     /*
3050      * Initialize for all unknown fields.
3051      */
3052     memset(headerBigBlock, 0, HEADER_SIZE);
3053 
3054     /*
3055      * Initialize the magic number.
3056      */
3057     memcpy(headerBigBlock, STORAGE_magic, sizeof(STORAGE_magic));
3058   }
3059 
3060   /*
3061    * Write the information to the header.
3062    */
3063   StorageUtl_WriteWord(
3064     headerBigBlock,
3065     OFFSET_MINORVERSION,
3066     0x3e);
3067 
3068   StorageUtl_WriteWord(
3069     headerBigBlock,
3070     OFFSET_MAJORVERSION,
3071     major_version);
3072 
3073   StorageUtl_WriteWord(
3074     headerBigBlock,
3075     OFFSET_BYTEORDERMARKER,
3076     (WORD)-2);
3077 
3078   StorageUtl_WriteWord(
3079     headerBigBlock,
3080     OFFSET_BIGBLOCKSIZEBITS,
3081     This->bigBlockSizeBits);
3082 
3083   StorageUtl_WriteWord(
3084     headerBigBlock,
3085     OFFSET_SMALLBLOCKSIZEBITS,
3086     This->smallBlockSizeBits);
3087 
3088   if (major_version >= 4)
3089   {
3090     if (This->rootBlockChain)
3091       dirsectorcount = BlockChainStream_GetCount(This->rootBlockChain);
3092     else
3093       /* This file is being created, and it will start out with one block. */
3094       dirsectorcount = 1;
3095   }
3096   else
3097     /* This field must be 0 in versions older than 4 */
3098     dirsectorcount = 0;
3099 
3100   StorageUtl_WriteDWord(
3101     headerBigBlock,
3102     OFFSET_DIRSECTORCOUNT,
3103     dirsectorcount);
3104 
3105   StorageUtl_WriteDWord(
3106     headerBigBlock,
3107     OFFSET_BBDEPOTCOUNT,
3108     This->bigBlockDepotCount);
3109 
3110   StorageUtl_WriteDWord(
3111     headerBigBlock,
3112     OFFSET_ROOTSTARTBLOCK,
3113     This->rootStartBlock);
3114 
3115   StorageUtl_WriteDWord(
3116     headerBigBlock,
3117     OFFSET_TRANSACTIONSIG,
3118     This->transactionSig);
3119 
3120   StorageUtl_WriteDWord(
3121     headerBigBlock,
3122     OFFSET_SMALLBLOCKLIMIT,
3123     This->smallBlockLimit);
3124 
3125   StorageUtl_WriteDWord(
3126     headerBigBlock,
3127     OFFSET_SBDEPOTSTART,
3128     This->smallBlockDepotStart);
3129 
3130   StorageUtl_WriteDWord(
3131     headerBigBlock,
3132     OFFSET_SBDEPOTCOUNT,
3133     This->smallBlockDepotChain ?
3134      BlockChainStream_GetCount(This->smallBlockDepotChain) : 0);
3135 
3136   StorageUtl_WriteDWord(
3137     headerBigBlock,
3138     OFFSET_EXTBBDEPOTSTART,
3139     This->extBigBlockDepotStart);
3140 
3141   StorageUtl_WriteDWord(
3142     headerBigBlock,
3143     OFFSET_EXTBBDEPOTCOUNT,
3144     This->extBigBlockDepotCount);
3145 
3146   for (index = 0; index < COUNT_BBDEPOTINHEADER; index ++)
3147   {
3148     StorageUtl_WriteDWord(
3149       headerBigBlock,
3150       OFFSET_BBDEPOTSTART + (sizeof(ULONG)*index),
3151       (This->bigBlockDepotStart[index]));
3152   }
3153 
3154   /*
3155    * Write the big block back to the file.
3156    */
3157   StorageImpl_WriteAt(This, offset, headerBigBlock, HEADER_SIZE, &bytes_written);
3158 }
3159 
3160 
3161 /************************************************************************
3162  * StorageImpl implementation : DirEntry methods
3163  ***********************************************************************/
3164 
3165 /******************************************************************************
3166  *      StorageImpl_ReadRawDirEntry
3167  *
3168  * This method will read the raw data from a directory entry in the file.
3169  *
3170  * buffer must be RAW_DIRENTRY_SIZE bytes long.
3171  */
3172 static HRESULT StorageImpl_ReadRawDirEntry(StorageImpl *This, ULONG index, BYTE *buffer)
3173 {
3174   ULARGE_INTEGER offset;
3175   HRESULT hr;
3176   ULONG bytesRead;
3177 
3178   offset.QuadPart  = (ULONGLONG)index * RAW_DIRENTRY_SIZE;
3179 
3180   hr = BlockChainStream_ReadAt(
3181                     This->rootBlockChain,
3182                     offset,
3183                     RAW_DIRENTRY_SIZE,
3184                     buffer,
3185                     &bytesRead);
3186 
3187   if (bytesRead != RAW_DIRENTRY_SIZE)
3188     return STG_E_READFAULT;
3189 
3190   return hr;
3191 }
3192 
3193 /******************************************************************************
3194  *      StorageImpl_WriteRawDirEntry
3195  *
3196  * This method will write the raw data from a directory entry in the file.
3197  *
3198  * buffer must be RAW_DIRENTRY_SIZE bytes long.
3199  */
3200 static HRESULT StorageImpl_WriteRawDirEntry(StorageImpl *This, ULONG index, const BYTE *buffer)
3201 {
3202   ULARGE_INTEGER offset;
3203   ULONG bytesRead;
3204 
3205   offset.QuadPart  = (ULONGLONG)index * RAW_DIRENTRY_SIZE;
3206 
3207   return BlockChainStream_WriteAt(
3208                     This->rootBlockChain,
3209                     offset,
3210                     RAW_DIRENTRY_SIZE,
3211                     buffer,
3212                     &bytesRead);
3213 }
3214 
3215 /***************************************************************************
3216  *
3217  * Internal Method
3218  *
3219  * Mark a directory entry in the file as free.
3220  */
3221 static HRESULT StorageImpl_DestroyDirEntry(
3222   StorageBaseImpl *base,
3223   DirRef index)
3224 {
3225   BYTE emptyData[RAW_DIRENTRY_SIZE];
3226   StorageImpl *storage = (StorageImpl*)base;
3227 
3228   memset(emptyData, 0, RAW_DIRENTRY_SIZE);
3229 
3230   return StorageImpl_WriteRawDirEntry(storage, index, emptyData);
3231 }
3232 
3233 /******************************************************************************
3234  *      UpdateRawDirEntry
3235  *
3236  * Update raw directory entry data from the fields in newData.
3237  *
3238  * buffer must be RAW_DIRENTRY_SIZE bytes long.
3239  */
3240 static void UpdateRawDirEntry(BYTE *buffer, const DirEntry *newData)
3241 {
3242   memset(buffer, 0, RAW_DIRENTRY_SIZE);
3243 
3244   memcpy(
3245     buffer + OFFSET_PS_NAME,
3246     newData->name,
3247     DIRENTRY_NAME_BUFFER_LEN );
3248 
3249   memcpy(buffer + OFFSET_PS_STGTYPE, &newData->stgType, 1);
3250 
3251   StorageUtl_WriteWord(
3252     buffer,
3253       OFFSET_PS_NAMELENGTH,
3254       newData->sizeOfNameString);
3255 
3256   StorageUtl_WriteDWord(
3257     buffer,
3258       OFFSET_PS_LEFTCHILD,
3259       newData->leftChild);
3260 
3261   StorageUtl_WriteDWord(
3262     buffer,
3263       OFFSET_PS_RIGHTCHILD,
3264       newData->rightChild);
3265 
3266   StorageUtl_WriteDWord(
3267     buffer,
3268       OFFSET_PS_DIRROOT,
3269       newData->dirRootEntry);
3270 
3271   StorageUtl_WriteGUID(
3272     buffer,
3273       OFFSET_PS_GUID,
3274       &newData->clsid);
3275 
3276   StorageUtl_WriteDWord(
3277     buffer,
3278       OFFSET_PS_CTIMELOW,
3279       newData->ctime.dwLowDateTime);
3280 
3281   StorageUtl_WriteDWord(
3282     buffer,
3283       OFFSET_PS_CTIMEHIGH,
3284       newData->ctime.dwHighDateTime);
3285 
3286   StorageUtl_WriteDWord(
3287     buffer,
3288       OFFSET_PS_MTIMELOW,
3289       newData->mtime.dwLowDateTime);
3290 
3291   StorageUtl_WriteDWord(
3292     buffer,
3293       OFFSET_PS_MTIMEHIGH,
3294       newData->ctime.dwHighDateTime);
3295 
3296   StorageUtl_WriteDWord(
3297     buffer,
3298       OFFSET_PS_STARTBLOCK,
3299       newData->startingBlock);
3300 
3301   StorageUtl_WriteDWord(
3302     buffer,
3303       OFFSET_PS_SIZE,
3304       newData->size.u.LowPart);
3305 
3306   StorageUtl_WriteDWord(
3307     buffer,
3308       OFFSET_PS_SIZE_HIGH,
3309       newData->size.u.HighPart);
3310 }
3311 
3312 /***************************************************************************
3313  *
3314  * Internal Method
3315  *
3316  * Reserve a directory entry in the file and initialize it.
3317  */
3318 static HRESULT StorageImpl_CreateDirEntry(
3319   StorageBaseImpl *base,
3320   const DirEntry *newData,
3321   DirRef *index)
3322 {
3323   StorageImpl *storage = (StorageImpl*)base;
3324   ULONG       currentEntryIndex    = 0;
3325   ULONG       newEntryIndex        = DIRENTRY_NULL;
3326   HRESULT hr = S_OK;
3327   BYTE currentData[RAW_DIRENTRY_SIZE];
3328   WORD sizeOfNameString;
3329 
3330   do
3331   {
3332     hr = StorageImpl_ReadRawDirEntry(storage,
3333                                      currentEntryIndex,
3334                                      currentData);
3335 
3336     if (SUCCEEDED(hr))
3337     {
3338       StorageUtl_ReadWord(
3339         currentData,
3340         OFFSET_PS_NAMELENGTH,
3341         &sizeOfNameString);
3342 
3343       if (sizeOfNameString == 0)
3344       {
3345         /*
3346          * The entry exists and is available, we found it.
3347          */
3348         newEntryIndex = currentEntryIndex;
3349       }
3350     }
3351     else
3352     {
3353       /*
3354        * We exhausted the directory entries, we will create more space below
3355        */
3356       newEntryIndex = currentEntryIndex;
3357     }
3358     currentEntryIndex++;
3359 
3360   } while (newEntryIndex == DIRENTRY_NULL);
3361 
3362   /*
3363    * grow the directory stream
3364    */
3365   if (FAILED(hr))
3366   {
3367     BYTE           emptyData[RAW_DIRENTRY_SIZE];
3368     ULARGE_INTEGER newSize;
3369     ULONG          entryIndex;
3370     ULONG          lastEntry     = 0;
3371     ULONG          blockCount    = 0;
3372 
3373     /*
3374      * obtain the new count of blocks in the directory stream
3375      */
3376     blockCount = BlockChainStream_GetCount(
3377                    storage->rootBlockChain)+1;
3378 
3379     /*
3380      * initialize the size used by the directory stream
3381      */
3382     newSize.QuadPart  = (ULONGLONG)storage->bigBlockSize * blockCount;
3383 
3384     /*
3385      * add a block to the directory stream
3386      */
3387     BlockChainStream_SetSize(storage->rootBlockChain, newSize);
3388 
3389     /*
3390      * memset the empty entry in order to initialize the unused newly
3391      * created entries
3392      */
3393     memset(emptyData, 0, RAW_DIRENTRY_SIZE);
3394 
3395     /*
3396      * initialize them
3397      */
3398     lastEntry = storage->bigBlockSize / RAW_DIRENTRY_SIZE * blockCount;
3399 
3400     for(
3401       entryIndex = newEntryIndex + 1;
3402       entryIndex < lastEntry;
3403       entryIndex++)
3404     {
3405       StorageImpl_WriteRawDirEntry(
3406         storage,
3407         entryIndex,
3408         emptyData);
3409     }
3410 
3411     StorageImpl_SaveFileHeader(storage);
3412   }
3413 
3414   UpdateRawDirEntry(currentData, newData);
3415 
3416   hr = StorageImpl_WriteRawDirEntry(storage, newEntryIndex, currentData);
3417 
3418   if (SUCCEEDED(hr))
3419     *index = newEntryIndex;
3420 
3421   return hr;
3422 }
3423 
3424 /******************************************************************************
3425  *      StorageImpl_ReadDirEntry
3426  *
3427  * This method will read the specified directory entry.
3428  */
3429 static HRESULT StorageImpl_ReadDirEntry(
3430   StorageImpl* This,
3431   DirRef         index,
3432   DirEntry*      buffer)
3433 {
3434   BYTE           currentEntry[RAW_DIRENTRY_SIZE];
3435   HRESULT        readRes;
3436 
3437   readRes = StorageImpl_ReadRawDirEntry(This, index, currentEntry);
3438 
3439   if (SUCCEEDED(readRes))
3440   {
3441     memset(buffer->name, 0, sizeof(buffer->name));
3442     memcpy(
3443       buffer->name,
3444       (WCHAR *)currentEntry+OFFSET_PS_NAME,
3445       DIRENTRY_NAME_BUFFER_LEN );
3446     TRACE("storage name: %s\n", debugstr_w(buffer->name));
3447 
3448     memcpy(&buffer->stgType, currentEntry + OFFSET_PS_STGTYPE, 1);
3449 
3450     StorageUtl_ReadWord(
3451       currentEntry,
3452       OFFSET_PS_NAMELENGTH,
3453       &buffer->sizeOfNameString);
3454 
3455     StorageUtl_ReadDWord(
3456       currentEntry,
3457       OFFSET_PS_LEFTCHILD,
3458       &buffer->leftChild);
3459 
3460     StorageUtl_ReadDWord(
3461       currentEntry,
3462       OFFSET_PS_RIGHTCHILD,
3463       &buffer->rightChild);
3464 
3465     StorageUtl_ReadDWord(
3466       currentEntry,
3467       OFFSET_PS_DIRROOT,
3468       &buffer->dirRootEntry);
3469 
3470     StorageUtl_ReadGUID(
3471       currentEntry,
3472       OFFSET_PS_GUID,
3473       &buffer->clsid);
3474 
3475     StorageUtl_ReadDWord(
3476       currentEntry,
3477       OFFSET_PS_CTIMELOW,
3478       &buffer->ctime.dwLowDateTime);
3479 
3480     StorageUtl_ReadDWord(
3481       currentEntry,
3482       OFFSET_PS_CTIMEHIGH,
3483       &buffer->ctime.dwHighDateTime);
3484 
3485     StorageUtl_ReadDWord(
3486       currentEntry,
3487       OFFSET_PS_MTIMELOW,
3488       &buffer->mtime.dwLowDateTime);
3489 
3490     StorageUtl_ReadDWord(
3491       currentEntry,
3492       OFFSET_PS_MTIMEHIGH,
3493       &buffer->mtime.dwHighDateTime);
3494 
3495     StorageUtl_ReadDWord(
3496       currentEntry,
3497       OFFSET_PS_STARTBLOCK,
3498       &buffer->startingBlock);
3499 
3500     StorageUtl_ReadDWord(
3501       currentEntry,
3502       OFFSET_PS_SIZE,
3503       &buffer->size.u.LowPart);
3504 
3505     if (This->bigBlockSize < 4096)
3506     {
3507       /* Version 3 files may have junk in the high part of size. */
3508       buffer->size.u.HighPart = 0;
3509     }
3510     else
3511     {
3512       StorageUtl_ReadDWord(
3513         currentEntry,
3514         OFFSET_PS_SIZE_HIGH,
3515         &buffer->size.u.HighPart);
3516     }
3517   }
3518 
3519   return readRes;
3520 }
3521 
3522 /*********************************************************************
3523  * Write the specified directory entry to the file
3524  */
3525 static HRESULT StorageImpl_WriteDirEntry(
3526   StorageImpl*          This,
3527   DirRef                index,
3528   const DirEntry*       buffer)
3529 {
3530   BYTE currentEntry[RAW_DIRENTRY_SIZE];
3531 
3532   UpdateRawDirEntry(currentEntry, buffer);
3533 
3534   return StorageImpl_WriteRawDirEntry(This, index, currentEntry);
3535 }
3536 
3537 
3538 /************************************************************************
3539  * StorageImpl implementation : Block methods
3540  ***********************************************************************/
3541 
3542 static ULONGLONG StorageImpl_GetBigBlockOffset(StorageImpl* This, ULONG index)
3543 {
3544     return (ULONGLONG)(index+1) * This->bigBlockSize;
3545 }
3546 
3547 static HRESULT StorageImpl_ReadBigBlock(
3548   StorageImpl* This,
3549   ULONG          blockIndex,
3550   void*          buffer,
3551   ULONG*         out_read)
3552 {
3553   ULARGE_INTEGER ulOffset;
3554   DWORD  read=0;
3555   HRESULT hr;
3556 
3557   ulOffset.QuadPart = StorageImpl_GetBigBlockOffset(This, blockIndex);
3558 
3559   hr = StorageImpl_ReadAt(This, ulOffset, buffer, This->bigBlockSize, &read);
3560 
3561   if (SUCCEEDED(hr) &&  read < This->bigBlockSize)
3562   {
3563     /* File ends during this block; fill the rest with 0's. */
3564     memset((LPBYTE)buffer+read, 0, This->bigBlockSize-read);
3565   }
3566 
3567   if (out_read) *out_read = read;
3568 
3569   return hr;
3570 }
3571 
3572 static BOOL StorageImpl_ReadDWordFromBigBlock(
3573   StorageImpl*  This,
3574   ULONG         blockIndex,
3575   ULONG         offset,
3576   DWORD*        value)
3577 {
3578   ULARGE_INTEGER ulOffset;
3579   DWORD  read;
3580   DWORD  tmp;
3581 
3582   ulOffset.QuadPart = StorageImpl_GetBigBlockOffset(This, blockIndex);
3583   ulOffset.QuadPart += offset;
3584 
3585   StorageImpl_ReadAt(This, ulOffset, &tmp, sizeof(DWORD), &read);
3586   *value = lendian32toh(tmp);
3587   return (read == sizeof(DWORD));
3588 }
3589 
3590 static BOOL StorageImpl_WriteBigBlock(
3591   StorageImpl*  This,
3592   ULONG         blockIndex,
3593   const void*   buffer)
3594 {
3595   ULARGE_INTEGER ulOffset;
3596   DWORD  wrote;
3597 
3598   ulOffset.QuadPart = StorageImpl_GetBigBlockOffset(This, blockIndex);
3599 
3600   StorageImpl_WriteAt(This, ulOffset, buffer, This->bigBlockSize, &wrote);
3601   return (wrote == This->bigBlockSize);
3602 }
3603 
3604 static BOOL StorageImpl_WriteDWordToBigBlock(
3605   StorageImpl* This,
3606   ULONG         blockIndex,
3607   ULONG         offset,
3608   DWORD         value)
3609 {
3610   ULARGE_INTEGER ulOffset;
3611   DWORD  wrote;
3612 
3613   ulOffset.QuadPart = StorageImpl_GetBigBlockOffset(This, blockIndex);
3614   ulOffset.QuadPart += offset;
3615 
3616   value = htole32(value);
3617   StorageImpl_WriteAt(This, ulOffset, &value, sizeof(DWORD), &wrote);
3618   return (wrote == sizeof(DWORD));
3619 }
3620 
3621 /******************************************************************************
3622  *              Storage32Impl_SmallBlocksToBigBlocks
3623  *
3624  * This method will convert a small block chain to a big block chain.
3625  * The small block chain will be destroyed.
3626  */
3627 static BlockChainStream* Storage32Impl_SmallBlocksToBigBlocks(
3628                       StorageImpl* This,
3629                       SmallBlockChainStream** ppsbChain)
3630 {
3631   ULONG bbHeadOfChain = BLOCK_END_OF_CHAIN;
3632   ULARGE_INTEGER size, offset;
3633   ULONG cbRead, cbWritten;
3634   ULARGE_INTEGER cbTotalRead;
3635   DirRef streamEntryRef;
3636   HRESULT resWrite = S_OK;
3637   HRESULT resRead;
3638   DirEntry streamEntry;
3639   BYTE *buffer;
3640   BlockChainStream *bbTempChain = NULL;
3641   BlockChainStream *bigBlockChain = NULL;
3642 
3643   /*
3644    * Create a temporary big block chain that doesn't have
3645    * an associated directory entry. This temporary chain will be
3646    * used to copy data from small blocks to big blocks.
3647    */
3648   bbTempChain = BlockChainStream_Construct(This,
3649                                            &bbHeadOfChain,
3650                                            DIRENTRY_NULL);
3651   if(!bbTempChain) return NULL;
3652   /*
3653    * Grow the big block chain.
3654    */
3655   size = SmallBlockChainStream_GetSize(*ppsbChain);
3656   BlockChainStream_SetSize(bbTempChain, size);
3657 
3658   /*
3659    * Copy the contents of the small block chain to the big block chain
3660    * by small block size increments.
3661    */
3662   offset.u.LowPart = 0;
3663   offset.u.HighPart = 0;
3664   cbTotalRead.QuadPart = 0;
3665 
3666   buffer = HeapAlloc(GetProcessHeap(),0,DEF_SMALL_BLOCK_SIZE);
3667   do
3668   {
3669     resRead = SmallBlockChainStream_ReadAt(*ppsbChain,
3670                                            offset,
3671                                            min(This->smallBlockSize, size.u.LowPart - offset.u.LowPart),
3672                                            buffer,
3673                                            &cbRead);
3674     if (FAILED(resRead))
3675         break;
3676 
3677     if (cbRead > 0)
3678     {
3679         cbTotalRead.QuadPart += cbRead;
3680 
3681         resWrite = BlockChainStream_WriteAt(bbTempChain,
3682                                             offset,
3683                                             cbRead,
3684                                             buffer,
3685                                             &cbWritten);
3686 
3687         if (FAILED(resWrite))
3688             break;
3689 
3690         offset.u.LowPart += cbRead;
3691     }
3692     else
3693     {
3694         resRead = STG_E_READFAULT;
3695         break;
3696     }
3697   } while (cbTotalRead.QuadPart < size.QuadPart);
3698   HeapFree(GetProcessHeap(),0,buffer);
3699 
3700   size.u.HighPart = 0;
3701   size.u.LowPart  = 0;
3702 
3703   if (FAILED(resRead) || FAILED(resWrite))
3704   {
3705     ERR("conversion failed: resRead = 0x%08x, resWrite = 0x%08x\n", resRead, resWrite);
3706     BlockChainStream_SetSize(bbTempChain, size);
3707     BlockChainStream_Destroy(bbTempChain);
3708     return NULL;
3709   }
3710 
3711   /*
3712    * Destroy the small block chain.
3713    */
3714   streamEntryRef = (*ppsbChain)->ownerDirEntry;
3715   SmallBlockChainStream_SetSize(*ppsbChain, size);
3716   SmallBlockChainStream_Destroy(*ppsbChain);
3717   *ppsbChain = 0;
3718 
3719   /*
3720    * Change the directory entry. This chain is now a big block chain
3721    * and it doesn't reside in the small blocks chain anymore.
3722    */
3723   StorageImpl_ReadDirEntry(This, streamEntryRef, &streamEntry);
3724 
3725   streamEntry.startingBlock = bbHeadOfChain;
3726 
3727   StorageImpl_WriteDirEntry(This, streamEntryRef, &streamEntry);
3728 
3729   /*
3730    * Destroy the temporary entryless big block chain.
3731    * Create a new big block chain associated with this entry.
3732    */
3733   BlockChainStream_Destroy(bbTempChain);
3734   bigBlockChain = BlockChainStream_Construct(This,
3735                                              NULL,
3736                                              streamEntryRef);
3737 
3738   return bigBlockChain;
3739 }
3740 
3741 /******************************************************************************
3742  *              Storage32Impl_BigBlocksToSmallBlocks
3743  *
3744  * This method will convert a big block chain to a small block chain.
3745  * The big block chain will be destroyed on success.
3746  */
3747 static SmallBlockChainStream* Storage32Impl_BigBlocksToSmallBlocks(
3748                            StorageImpl* This,
3749                            BlockChainStream** ppbbChain,
3750                            ULARGE_INTEGER newSize)
3751 {
3752     ULARGE_INTEGER size, offset, cbTotalRead;
3753     ULONG cbRead, cbWritten, sbHeadOfChain = BLOCK_END_OF_CHAIN;
3754     DirRef streamEntryRef;
3755     HRESULT resWrite = S_OK, resRead = S_OK;
3756     DirEntry streamEntry;
3757     BYTE* buffer;
3758     SmallBlockChainStream* sbTempChain;
3759 
3760     TRACE("%p %p\n", This, ppbbChain);
3761 
3762     sbTempChain = SmallBlockChainStream_Construct(This, &sbHeadOfChain,
3763             DIRENTRY_NULL);
3764 
3765     if(!sbTempChain)
3766         return NULL;
3767 
3768     SmallBlockChainStream_SetSize(sbTempChain, newSize);
3769     size = BlockChainStream_GetSize(*ppbbChain);
3770     size.QuadPart = min(size.QuadPart, newSize.QuadPart);
3771 
3772     offset.u.HighPart = 0;
3773     offset.u.LowPart = 0;
3774     cbTotalRead.QuadPart = 0;
3775     buffer = HeapAlloc(GetProcessHeap(), 0, This->bigBlockSize);
3776     while(cbTotalRead.QuadPart < size.QuadPart)
3777     {
3778         resRead = BlockChainStream_ReadAt(*ppbbChain, offset,
3779                 min(This->bigBlockSize, size.u.LowPart - offset.u.LowPart),
3780                 buffer, &cbRead);
3781 
3782         if(FAILED(resRead))
3783             break;
3784 
3785         if(cbRead > 0)
3786         {
3787             cbTotalRead.QuadPart += cbRead;
3788 
3789             resWrite = SmallBlockChainStream_WriteAt(sbTempChain, offset,
3790                     cbRead, buffer, &cbWritten);
3791 
3792             if(FAILED(resWrite))
3793                 break;
3794 
3795             offset.u.LowPart += cbRead;
3796         }
3797         else
3798         {
3799             resRead = STG_E_READFAULT;
3800             break;
3801         }
3802     }
3803     HeapFree(GetProcessHeap(), 0, buffer);
3804 
3805     size.u.HighPart = 0;
3806     size.u.LowPart = 0;
3807 
3808     if(FAILED(resRead) || FAILED(resWrite))
3809     {
3810         ERR("conversion failed: resRead = 0x%08x, resWrite = 0x%08x\n", resRead, resWrite);
3811         SmallBlockChainStream_SetSize(sbTempChain, size);
3812         SmallBlockChainStream_Destroy(sbTempChain);
3813         return NULL;
3814     }
3815 
3816     /* destroy the original big block chain */
3817     streamEntryRef = (*ppbbChain)->ownerDirEntry;
3818     BlockChainStream_SetSize(*ppbbChain, size);
3819     BlockChainStream_Destroy(*ppbbChain);
3820     *ppbbChain = NULL;
3821 
3822     StorageImpl_ReadDirEntry(This, streamEntryRef, &streamEntry);
3823     streamEntry.startingBlock = sbHeadOfChain;
3824     StorageImpl_WriteDirEntry(This, streamEntryRef, &streamEntry);
3825 
3826     SmallBlockChainStream_Destroy(sbTempChain);
3827     return SmallBlockChainStream_Construct(This, NULL, streamEntryRef);
3828 }
3829 
3830 /******************************************************************************
3831  *      Storage32Impl_AddBlockDepot
3832  *
3833  * This will create a depot block, essentially it is a block initialized
3834  * to BLOCK_UNUSEDs.
3835  */
3836 static void Storage32Impl_AddBlockDepot(StorageImpl* This, ULONG blockIndex, ULONG depotIndex)
3837 {
3838   BYTE blockBuffer[MAX_BIG_BLOCK_SIZE];
3839   ULONG rangeLockIndex = RANGELOCK_FIRST / This->bigBlockSize - 1;
3840   ULONG blocksPerDepot = This->bigBlockSize / sizeof(ULONG);
3841   ULONG rangeLockDepot = rangeLockIndex / blocksPerDepot;
3842 
3843   /*
3844    * Initialize blocks as free
3845    */
3846   memset(blockBuffer, BLOCK_UNUSED, This->bigBlockSize);
3847 
3848   /* Reserve the range lock sector */
3849   if (depotIndex == rangeLockDepot)
3850   {
3851     ((ULONG*)blockBuffer)[rangeLockIndex % blocksPerDepot] = BLOCK_END_OF_CHAIN;
3852   }
3853 
3854   StorageImpl_WriteBigBlock(This, blockIndex, blockBuffer);
3855 }
3856 
3857 /******************************************************************************
3858  *      Storage32Impl_GetExtDepotBlock
3859  *
3860  * Returns the index of the block that corresponds to the specified depot
3861  * index. This method is only for depot indexes equal or greater than
3862  * COUNT_BBDEPOTINHEADER.
3863  */
3864 static ULONG Storage32Impl_GetExtDepotBlock(StorageImpl* This, ULONG depotIndex)
3865 {
3866   ULONG depotBlocksPerExtBlock = (This->bigBlockSize / sizeof(ULONG)) - 1;
3867   ULONG numExtBlocks           = depotIndex - COUNT_BBDEPOTINHEADER;
3868   ULONG extBlockCount          = numExtBlocks / depotBlocksPerExtBlock;
3869   ULONG extBlockOffset         = numExtBlocks % depotBlocksPerExtBlock;
3870   ULONG blockIndex             = BLOCK_UNUSED;
3871   ULONG extBlockIndex;
3872   BYTE depotBuffer[MAX_BIG_BLOCK_SIZE];
3873   int index, num_blocks;
3874 
3875   assert(depotIndex >= COUNT_BBDEPOTINHEADER);
3876 
3877   if (extBlockCount >= This->extBigBlockDepotCount)
3878     return BLOCK_UNUSED;
3879 
3880   if (This->indexExtBlockDepotCached != extBlockCount)
3881   {
3882     extBlockIndex = This->extBigBlockDepotLocations[extBlockCount];
3883 
3884     StorageImpl_ReadBigBlock(This, extBlockIndex, depotBuffer, NULL);
3885 
3886     num_blocks = This->bigBlockSize / 4;
3887 
3888     for (index = 0; index < num_blocks; index++)
3889     {
3890       StorageUtl_ReadDWord(depotBuffer, index*sizeof(ULONG), &blockIndex);
3891       This->extBlockDepotCached[index] = blockIndex;
3892     }
3893 
3894     This->indexExtBlockDepotCached = extBlockCount;
3895   }
3896 
3897   blockIndex = This->extBlockDepotCached[extBlockOffset];
3898 
3899   return blockIndex;
3900 }
3901 
3902 /******************************************************************************
3903  *      Storage32Impl_SetExtDepotBlock
3904  *
3905  * Associates the specified block index to the specified depot index.
3906  * This method is only for depot indexes equal or greater than
3907  * COUNT_BBDEPOTINHEADER.
3908  */
3909 static void Storage32Impl_SetExtDepotBlock(StorageImpl* This, ULONG depotIndex, ULONG blockIndex)
3910 {
3911   ULONG depotBlocksPerExtBlock = (This->bigBlockSize / sizeof(ULONG)) - 1;
3912   ULONG numExtBlocks           = depotIndex - COUNT_BBDEPOTINHEADER;
3913   ULONG extBlockCount          = numExtBlocks / depotBlocksPerExtBlock;
3914   ULONG extBlockOffset         = numExtBlocks % depotBlocksPerExtBlock;
3915   ULONG extBlockIndex;
3916 
3917   assert(depotIndex >= COUNT_BBDEPOTINHEADER);
3918 
3919   assert(extBlockCount < This->extBigBlockDepotCount);
3920 
3921   extBlockIndex = This->extBigBlockDepotLocations[extBlockCount];
3922 
3923   if (extBlockIndex != BLOCK_UNUSED)
3924   {
3925     StorageImpl_WriteDWordToBigBlock(This, extBlockIndex,
3926                         extBlockOffset * sizeof(ULONG),
3927                         blockIndex);
3928   }
3929 
3930   if (This->indexExtBlockDepotCached == extBlockCount)
3931   {
3932     This->extBlockDepotCached[extBlockOffset] = blockIndex;
3933   }
3934 }
3935 
3936 /******************************************************************************
3937  *      Storage32Impl_AddExtBlockDepot
3938  *
3939  * Creates an extended depot block.
3940  */
3941 static ULONG Storage32Impl_AddExtBlockDepot(StorageImpl* This)
3942 {
3943   ULONG numExtBlocks           = This->extBigBlockDepotCount;
3944   ULONG nextExtBlock           = This->extBigBlockDepotStart;
3945   BYTE  depotBuffer[MAX_BIG_BLOCK_SIZE];
3946   ULONG index                  = BLOCK_UNUSED;
3947   ULONG nextBlockOffset        = This->bigBlockSize - sizeof(ULONG);
3948   ULONG blocksPerDepotBlock    = This->bigBlockSize / sizeof(ULONG);
3949   ULONG depotBlocksPerExtBlock = blocksPerDepotBlock - 1;
3950 
3951   index = (COUNT_BBDEPOTINHEADER + (numExtBlocks * depotBlocksPerExtBlock)) *
3952           blocksPerDepotBlock;
3953 
3954   if ((numExtBlocks == 0) && (nextExtBlock == BLOCK_END_OF_CHAIN))
3955   {
3956     /*
3957      * The first extended block.
3958      */
3959     This->extBigBlockDepotStart = index;
3960   }
3961   else
3962   {
3963     /*
3964      * Find the last existing extended block.
3965      */
3966     nextExtBlock = This->extBigBlockDepotLocations[This->extBigBlockDepotCount-1];
3967 
3968     /*
3969      * Add the new extended block to the chain.
3970      */
3971     StorageImpl_WriteDWordToBigBlock(This, nextExtBlock, nextBlockOffset,
3972                                      index);
3973   }
3974 
3975   /*
3976    * Initialize this block.
3977    */
3978   memset(depotBuffer, BLOCK_UNUSED, This->bigBlockSize);
3979   StorageImpl_WriteBigBlock(This, index, depotBuffer);
3980 
3981   /* Add the block to our cache. */
3982   if (This->extBigBlockDepotLocationsSize == numExtBlocks)
3983   {
3984     ULONG new_cache_size = (This->extBigBlockDepotLocationsSize+1)*2;
3985     ULONG *new_cache = HeapAlloc(GetProcessHeap(), 0, sizeof(ULONG) * new_cache_size);
3986 
3987     memcpy(new_cache, This->extBigBlockDepotLocations, sizeof(ULONG) * This->extBigBlockDepotLocationsSize);
3988     HeapFree(GetProcessHeap(), 0, This->extBigBlockDepotLocations);
3989 
3990     This->extBigBlockDepotLocations = new_cache;
3991     This->extBigBlockDepotLocationsSize = new_cache_size;
3992   }
3993   This->extBigBlockDepotLocations[numExtBlocks] = index;
3994 
3995   return index;
3996 }
3997 
3998 /************************************************************************
3999  * StorageImpl_GetNextBlockInChain
4000  *
4001  * This method will retrieve the block index of the next big block in
4002  * in the chain.
4003  *
4004  * Params:  This       - Pointer to the Storage object.
4005  *          blockIndex - Index of the block to retrieve the chain
4006  *                       for.
4007  *          nextBlockIndex - receives the return value.
4008  *
4009  * Returns: This method returns the index of the next block in the chain.
4010  *          It will return the constants:
4011  *              BLOCK_SPECIAL - If the block given was not part of a
4012  *                              chain.
4013  *              BLOCK_END_OF_CHAIN - If the block given was the last in
4014  *                                   a chain.
4015  *              BLOCK_UNUSED - If the block given was not past of a chain
4016  *                             and is available.
4017  *              BLOCK_EXTBBDEPOT - This block is part of the extended
4018  *                                 big block depot.
4019  *
4020  * See Windows documentation for more details on IStorage methods.
4021  */
4022 static HRESULT StorageImpl_GetNextBlockInChain(
4023   StorageImpl* This,
4024   ULONG        blockIndex,
4025   ULONG*       nextBlockIndex)
4026 {
4027   ULONG offsetInDepot    = blockIndex * sizeof (ULONG);
4028   ULONG depotBlockCount  = offsetInDepot / This->bigBlockSize;
4029   ULONG depotBlockOffset = offsetInDepot % This->bigBlockSize;
4030   BYTE depotBuffer[MAX_BIG_BLOCK_SIZE];
4031   ULONG read;
4032   ULONG depotBlockIndexPos;
4033   int index, num_blocks;
4034 
4035   *nextBlockIndex   = BLOCK_SPECIAL;
4036 
4037   if(depotBlockCount >= This->bigBlockDepotCount)
4038   {
4039     WARN("depotBlockCount %d, bigBlockDepotCount %d\n", depotBlockCount,
4040 	 This->bigBlockDepotCount);
4041     return STG_E_READFAULT;
4042   }
4043 
4044   /*
4045    * Cache the currently accessed depot block.
4046    */
4047   if (depotBlockCount != This->indexBlockDepotCached)
4048   {
4049     This->indexBlockDepotCached = depotBlockCount;
4050 
4051     if (depotBlockCount < COUNT_BBDEPOTINHEADER)
4052     {
4053       depotBlockIndexPos = This->bigBlockDepotStart[depotBlockCount];
4054     }
4055     else
4056     {
4057       /*
4058        * We have to look in the extended depot.
4059        */
4060       depotBlockIndexPos = Storage32Impl_GetExtDepotBlock(This, depotBlockCount);
4061     }
4062 
4063     StorageImpl_ReadBigBlock(This, depotBlockIndexPos, depotBuffer, &read);
4064 
4065     if (!read)
4066       return STG_E_READFAULT;
4067 
4068     num_blocks = This->bigBlockSize / 4;
4069 
4070     for (index = 0; index < num_blocks; index++)
4071     {
4072       StorageUtl_ReadDWord(depotBuffer, index*sizeof(ULONG), nextBlockIndex);
4073       This->blockDepotCached[index] = *nextBlockIndex;
4074     }
4075   }
4076 
4077   *nextBlockIndex = This->blockDepotCached[depotBlockOffset/sizeof(ULONG)];
4078 
4079   return S_OK;
4080 }
4081 
4082 /******************************************************************************
4083  *      Storage32Impl_GetNextExtendedBlock
4084  *
4085  * Given an extended block this method will return the next extended block.
4086  *
4087  * NOTES:
4088  * The last ULONG of an extended block is the block index of the next
4089  * extended block. Extended blocks are marked as BLOCK_EXTBBDEPOT in the
4090  * depot.
4091  *
4092  * Return values:
4093  *    - The index of the next extended block
4094  *    - BLOCK_UNUSED: there is no next extended block.
4095  *    - Any other return values denotes failure.
4096  */
4097 static ULONG Storage32Impl_GetNextExtendedBlock(StorageImpl* This, ULONG blockIndex)
4098 {
4099   ULONG nextBlockIndex   = BLOCK_SPECIAL;
4100   ULONG depotBlockOffset = This->bigBlockSize - sizeof(ULONG);
4101 
4102   StorageImpl_ReadDWordFromBigBlock(This, blockIndex, depotBlockOffset,
4103                         &nextBlockIndex);
4104 
4105   return nextBlockIndex;
4106 }
4107 
4108 /******************************************************************************
4109  *      StorageImpl_SetNextBlockInChain
4110  *
4111  * This method will write the index of the specified block's next block
4112  * in the big block depot.
4113  *
4114  * For example: to create the chain 3 -> 1 -> 7 -> End of Chain
4115  *              do the following
4116  *
4117  * StorageImpl_SetNextBlockInChain(This, 3, 1);
4118  * StorageImpl_SetNextBlockInChain(This, 1, 7);
4119  * StorageImpl_SetNextBlockInChain(This, 7, BLOCK_END_OF_CHAIN);
4120  *
4121  */
4122 static void StorageImpl_SetNextBlockInChain(
4123           StorageImpl* This,
4124           ULONG          blockIndex,
4125           ULONG          nextBlock)
4126 {
4127   ULONG offsetInDepot    = blockIndex * sizeof (ULONG);
4128   ULONG depotBlockCount  = offsetInDepot / This->bigBlockSize;
4129   ULONG depotBlockOffset = offsetInDepot % This->bigBlockSize;
4130   ULONG depotBlockIndexPos;
4131 
4132   assert(depotBlockCount < This->bigBlockDepotCount);
4133   assert(blockIndex != nextBlock);
4134 
4135   if (blockIndex == (RANGELOCK_FIRST / This->bigBlockSize) - 1)
4136     /* This should never happen (storage file format spec forbids it), but
4137      * older versions of Wine may have generated broken files. We don't want to
4138      * assert and potentially lose data, but we do want to know if this ever
4139      * happens in a newly-created file. */
4140     ERR("Using range lock page\n");
4141 
4142   if (depotBlockCount < COUNT_BBDEPOTINHEADER)
4143   {
4144     depotBlockIndexPos = This->bigBlockDepotStart[depotBlockCount];
4145   }
4146   else
4147   {
4148     /*
4149      * We have to look in the extended depot.
4150      */
4151     depotBlockIndexPos = Storage32Impl_GetExtDepotBlock(This, depotBlockCount);
4152   }
4153 
4154   StorageImpl_WriteDWordToBigBlock(This, depotBlockIndexPos, depotBlockOffset,
4155                         nextBlock);
4156   /*
4157    * Update the cached block depot, if necessary.
4158    */
4159   if (depotBlockCount == This->indexBlockDepotCached)
4160   {
4161     This->blockDepotCached[depotBlockOffset/sizeof(ULONG)] = nextBlock;
4162   }
4163 }
4164 
4165 /******************************************************************************
4166  *      StorageImpl_GetNextFreeBigBlock
4167  *
4168  * Returns the index of the next free big block.
4169  * If the big block depot is filled, this method will enlarge it.
4170  *
4171  */
4172 static ULONG StorageImpl_GetNextFreeBigBlock(
4173   StorageImpl* This)
4174 {
4175   ULONG depotBlockIndexPos;
4176   BYTE depotBuffer[MAX_BIG_BLOCK_SIZE];
4177   ULONG depotBlockOffset;
4178   ULONG blocksPerDepot    = This->bigBlockSize / sizeof(ULONG);
4179   ULONG nextBlockIndex    = BLOCK_SPECIAL;
4180   int   depotIndex        = 0;
4181   ULONG freeBlock         = BLOCK_UNUSED;
4182   ULONG read;
4183   ULARGE_INTEGER neededSize;
4184   STATSTG statstg;
4185 
4186   depotIndex = This->prevFreeBlock / blocksPerDepot;
4187   depotBlockOffset = (This->prevFreeBlock % blocksPerDepot) * sizeof(ULONG);
4188 
4189   /*
4190    * Scan the entire big block depot until we find a block marked free
4191    */
4192   while (nextBlockIndex != BLOCK_UNUSED)
4193   {
4194     if (depotIndex < COUNT_BBDEPOTINHEADER)
4195     {
4196       depotBlockIndexPos = This->bigBlockDepotStart[depotIndex];
4197 
4198       /*
4199        * Grow the primary depot.
4200        */
4201       if (depotBlockIndexPos == BLOCK_UNUSED)
4202       {
4203         depotBlockIndexPos = depotIndex*blocksPerDepot;
4204 
4205         /*
4206          * Add a block depot.
4207          */
4208         Storage32Impl_AddBlockDepot(This, depotBlockIndexPos, depotIndex);
4209         This->bigBlockDepotCount++;
4210         This->bigBlockDepotStart[depotIndex] = depotBlockIndexPos;
4211 
4212         /*
4213          * Flag it as a block depot.
4214          */
4215         StorageImpl_SetNextBlockInChain(This,
4216                                           depotBlockIndexPos,
4217                                           BLOCK_SPECIAL);
4218 
4219         /* Save new header information.
4220          */
4221         StorageImpl_SaveFileHeader(This);
4222       }
4223     }
4224     else
4225     {
4226       depotBlockIndexPos = Storage32Impl_GetExtDepotBlock(This, depotIndex);
4227 
4228       if (depotBlockIndexPos == BLOCK_UNUSED)
4229       {
4230         /*
4231          * Grow the extended depot.
4232          */
4233         ULONG extIndex       = BLOCK_UNUSED;
4234         ULONG numExtBlocks   = depotIndex - COUNT_BBDEPOTINHEADER;
4235         ULONG extBlockOffset = numExtBlocks % (blocksPerDepot - 1);
4236 
4237         if (extBlockOffset == 0)
4238         {
4239           /* We need an extended block.
4240            */
4241           extIndex = Storage32Impl_AddExtBlockDepot(This);
4242           This->extBigBlockDepotCount++;
4243           depotBlockIndexPos = extIndex + 1;
4244         }
4245         else
4246           depotBlockIndexPos = depotIndex * blocksPerDepot;
4247 
4248         /*
4249          * Add a block depot and mark it in the extended block.
4250          */
4251         Storage32Impl_AddBlockDepot(This, depotBlockIndexPos, depotIndex);
4252         This->bigBlockDepotCount++;
4253         Storage32Impl_SetExtDepotBlock(This, depotIndex, depotBlockIndexPos);
4254 
4255         /* Flag the block depot.
4256          */
4257         StorageImpl_SetNextBlockInChain(This,
4258                                           depotBlockIndexPos,
4259                                           BLOCK_SPECIAL);
4260 
4261         /* If necessary, flag the extended depot block.
4262          */
4263         if (extIndex != BLOCK_UNUSED)
4264           StorageImpl_SetNextBlockInChain(This, extIndex, BLOCK_EXTBBDEPOT);
4265 
4266         /* Save header information.
4267          */
4268         StorageImpl_SaveFileHeader(This);
4269       }
4270     }
4271 
4272     StorageImpl_ReadBigBlock(This, depotBlockIndexPos, depotBuffer, &read);
4273 
4274     if (read)
4275     {
4276       while ( ( (depotBlockOffset/sizeof(ULONG) ) < blocksPerDepot) &&
4277               ( nextBlockIndex != BLOCK_UNUSED))
4278       {
4279         StorageUtl_ReadDWord(depotBuffer, depotBlockOffset, &nextBlockIndex);
4280 
4281         if (nextBlockIndex == BLOCK_UNUSED)
4282         {
4283           freeBlock = (depotIndex * blocksPerDepot) +
4284                       (depotBlockOffset/sizeof(ULONG));
4285         }
4286 
4287         depotBlockOffset += sizeof(ULONG);
4288       }
4289     }
4290 
4291     depotIndex++;
4292     depotBlockOffset = 0;
4293   }
4294 
4295   /*
4296    * make sure that the block physically exists before using it
4297    */
4298   neededSize.QuadPart = StorageImpl_GetBigBlockOffset(This, freeBlock)+This->bigBlockSize;
4299 
4300   ILockBytes_Stat(This->lockBytes, &statstg, STATFLAG_NONAME);
4301 
4302   if (neededSize.QuadPart > statstg.cbSize.QuadPart)
4303     ILockBytes_SetSize(This->lockBytes, neededSize);
4304 
4305   This->prevFreeBlock = freeBlock;
4306 
4307   return freeBlock;
4308 }
4309 
4310 /******************************************************************************
4311  *      StorageImpl_FreeBigBlock
4312  *
4313  * This method will flag the specified block as free in the big block depot.
4314  */
4315 static void StorageImpl_FreeBigBlock(
4316   StorageImpl* This,
4317   ULONG          blockIndex)
4318 {
4319   StorageImpl_SetNextBlockInChain(This, blockIndex, BLOCK_UNUSED);
4320 
4321   if (blockIndex < This->prevFreeBlock)
4322     This->prevFreeBlock = blockIndex;
4323 }
4324 
4325 
4326 static HRESULT StorageImpl_BaseWriteDirEntry(StorageBaseImpl *base,
4327   DirRef index, const DirEntry *data)
4328 {
4329   StorageImpl *This = (StorageImpl*)base;
4330   return StorageImpl_WriteDirEntry(This, index, data);
4331 }
4332 
4333 static HRESULT StorageImpl_BaseReadDirEntry(StorageBaseImpl *base,
4334   DirRef index, DirEntry *data)
4335 {
4336   StorageImpl *This = (StorageImpl*)base;
4337   return StorageImpl_ReadDirEntry(This, index, data);
4338 }
4339 
4340 static BlockChainStream **StorageImpl_GetFreeBlockChainCacheEntry(StorageImpl* This)
4341 {
4342   int i;
4343 
4344   for (i=0; i<BLOCKCHAIN_CACHE_SIZE; i++)
4345   {
4346     if (!This->blockChainCache[i])
4347     {
4348       return &This->blockChainCache[i];
4349     }
4350   }
4351 
4352   i = This->blockChainToEvict;
4353 
4354   BlockChainStream_Destroy(This->blockChainCache[i]);
4355   This->blockChainCache[i] = NULL;
4356 
4357   This->blockChainToEvict++;
4358   if (This->blockChainToEvict == BLOCKCHAIN_CACHE_SIZE)
4359     This->blockChainToEvict = 0;
4360 
4361   return &This->blockChainCache[i];
4362 }
4363 
4364 static BlockChainStream **StorageImpl_GetCachedBlockChainStream(StorageImpl *This,
4365     DirRef index)
4366 {
4367   int i, free_index=-1;
4368 
4369   for (i=0; i<BLOCKCHAIN_CACHE_SIZE; i++)
4370   {
4371     if (!This->blockChainCache[i])
4372     {
4373       if (free_index == -1) free_index = i;
4374     }
4375     else if (This->blockChainCache[i]->ownerDirEntry == index)
4376     {
4377       return &This->blockChainCache[i];
4378     }
4379   }
4380 
4381   if (free_index == -1)
4382   {
4383     free_index = This->blockChainToEvict;
4384 
4385     BlockChainStream_Destroy(This->blockChainCache[free_index]);
4386     This->blockChainCache[free_index] = NULL;
4387 
4388     This->blockChainToEvict++;
4389     if (This->blockChainToEvict == BLOCKCHAIN_CACHE_SIZE)
4390       This->blockChainToEvict = 0;
4391   }
4392 
4393   This->blockChainCache[free_index] = BlockChainStream_Construct(This, NULL, index);
4394   return &This->blockChainCache[free_index];
4395 }
4396 
4397 static void StorageImpl_DeleteCachedBlockChainStream(StorageImpl *This, DirRef index)
4398 {
4399   int i;
4400 
4401   for (i=0; i<BLOCKCHAIN_CACHE_SIZE; i++)
4402   {
4403     if (This->blockChainCache[i] && This->blockChainCache[i]->ownerDirEntry == index)
4404     {
4405       BlockChainStream_Destroy(This->blockChainCache[i]);
4406       This->blockChainCache[i] = NULL;
4407       return;
4408     }
4409   }
4410 }
4411 
4412 static HRESULT StorageImpl_StreamReadAt(StorageBaseImpl *base, DirRef index,
4413   ULARGE_INTEGER offset, ULONG size, void *buffer, ULONG *bytesRead)
4414 {
4415   StorageImpl *This = (StorageImpl*)base;
4416   DirEntry data;
4417   HRESULT hr;
4418   ULONG bytesToRead;
4419 
4420   hr = StorageImpl_ReadDirEntry(This, index, &data);
4421   if (FAILED(hr)) return hr;
4422 
4423   if (data.size.QuadPart == 0)
4424   {
4425     *bytesRead = 0;
4426     return S_OK;
4427   }
4428 
4429   if (offset.QuadPart + size > data.size.QuadPart)
4430   {
4431     bytesToRead = data.size.QuadPart - offset.QuadPart;
4432   }
4433   else
4434   {
4435     bytesToRead = size;
4436   }
4437 
4438   if (data.size.QuadPart < LIMIT_TO_USE_SMALL_BLOCK)
4439   {
4440     SmallBlockChainStream *stream;
4441 
4442     stream = SmallBlockChainStream_Construct(This, NULL, index);
4443     if (!stream) return E_OUTOFMEMORY;
4444 
4445     hr = SmallBlockChainStream_ReadAt(stream, offset, bytesToRead, buffer, bytesRead);
4446 
4447     SmallBlockChainStream_Destroy(stream);
4448 
4449     return hr;
4450   }
4451   else
4452   {
4453     BlockChainStream *stream = NULL;
4454 
4455     stream = *StorageImpl_GetCachedBlockChainStream(This, index);
4456     if (!stream) return E_OUTOFMEMORY;
4457 
4458     hr = BlockChainStream_ReadAt(stream, offset, bytesToRead, buffer, bytesRead);
4459 
4460     return hr;
4461   }
4462 }
4463 
4464 static HRESULT StorageImpl_StreamSetSize(StorageBaseImpl *base, DirRef index,
4465   ULARGE_INTEGER newsize)
4466 {
4467   StorageImpl *This = (StorageImpl*)base;
4468   DirEntry data;
4469   HRESULT hr;
4470   SmallBlockChainStream *smallblock=NULL;
4471   BlockChainStream **pbigblock=NULL, *bigblock=NULL;
4472 
4473   hr = StorageImpl_ReadDirEntry(This, index, &data);
4474   if (FAILED(hr)) return hr;
4475 
4476   /* In simple mode keep the stream size above the small block limit */
4477   if (This->base.openFlags & STGM_SIMPLE)
4478     newsize.QuadPart = max(newsize.QuadPart, LIMIT_TO_USE_SMALL_BLOCK);
4479 
4480   if (data.size.QuadPart == newsize.QuadPart)
4481     return S_OK;
4482 
4483   /* Create a block chain object of the appropriate type */
4484   if (data.size.QuadPart == 0)
4485   {
4486     if (newsize.QuadPart < LIMIT_TO_USE_SMALL_BLOCK)
4487     {
4488       smallblock = SmallBlockChainStream_Construct(This, NULL, index);
4489       if (!smallblock) return E_OUTOFMEMORY;
4490     }
4491     else
4492     {
4493       pbigblock = StorageImpl_GetCachedBlockChainStream(This, index);
4494       bigblock = *pbigblock;
4495       if (!bigblock) return E_OUTOFMEMORY;
4496     }
4497   }
4498   else if (data.size.QuadPart < LIMIT_TO_USE_SMALL_BLOCK)
4499   {
4500     smallblock = SmallBlockChainStream_Construct(This, NULL, index);
4501     if (!smallblock) return E_OUTOFMEMORY;
4502   }
4503   else
4504   {
4505     pbigblock = StorageImpl_GetCachedBlockChainStream(This, index);
4506     bigblock = *pbigblock;
4507     if (!bigblock) return E_OUTOFMEMORY;
4508   }
4509 
4510   /* Change the block chain type if necessary. */
4511   if (smallblock && newsize.QuadPart >= LIMIT_TO_USE_SMALL_BLOCK)
4512   {
4513     bigblock = Storage32Impl_SmallBlocksToBigBlocks(This, &smallblock);
4514     if (!bigblock)
4515     {
4516       SmallBlockChainStream_Destroy(smallblock);
4517       return E_FAIL;
4518     }
4519 
4520     pbigblock = StorageImpl_GetFreeBlockChainCacheEntry(This);
4521     *pbigblock = bigblock;
4522   }
4523   else if (bigblock && newsize.QuadPart < LIMIT_TO_USE_SMALL_BLOCK)
4524   {
4525     smallblock = Storage32Impl_BigBlocksToSmallBlocks(This, pbigblock, newsize);
4526     if (!smallblock)
4527       return E_FAIL;
4528   }
4529 
4530   /* Set the size of the block chain. */
4531   if (smallblock)
4532   {
4533     SmallBlockChainStream_SetSize(smallblock, newsize);
4534     SmallBlockChainStream_Destroy(smallblock);
4535   }
4536   else
4537   {
4538     BlockChainStream_SetSize(bigblock, newsize);
4539   }
4540 
4541   /* Set the size in the directory entry. */
4542   hr = StorageImpl_ReadDirEntry(This, index, &data);
4543   if (SUCCEEDED(hr))
4544   {
4545     data.size = newsize;
4546 
4547     hr = StorageImpl_WriteDirEntry(This, index, &data);
4548   }
4549   return hr;
4550 }
4551 
4552 static HRESULT StorageImpl_StreamWriteAt(StorageBaseImpl *base, DirRef index,
4553   ULARGE_INTEGER offset, ULONG size, const void *buffer, ULONG *bytesWritten)
4554 {
4555   StorageImpl *This = (StorageImpl*)base;
4556   DirEntry data;
4557   HRESULT hr;
4558   ULARGE_INTEGER newSize;
4559 
4560   hr = StorageImpl_ReadDirEntry(This, index, &data);
4561   if (FAILED(hr)) return hr;
4562 
4563   /* Grow the stream if necessary */
4564   newSize.QuadPart = offset.QuadPart + size;
4565 
4566   if (newSize.QuadPart > data.size.QuadPart)
4567   {
4568     hr = StorageImpl_StreamSetSize(base, index, newSize);
4569     if (FAILED(hr))
4570       return hr;
4571 
4572     hr = StorageImpl_ReadDirEntry(This, index, &data);
4573     if (FAILED(hr)) return hr;
4574   }
4575 
4576   if (data.size.QuadPart < LIMIT_TO_USE_SMALL_BLOCK)
4577   {
4578     SmallBlockChainStream *stream;
4579 
4580     stream = SmallBlockChainStream_Construct(This, NULL, index);
4581     if (!stream) return E_OUTOFMEMORY;
4582 
4583     hr = SmallBlockChainStream_WriteAt(stream, offset, size, buffer, bytesWritten);
4584 
4585     SmallBlockChainStream_Destroy(stream);
4586 
4587     return hr;
4588   }
4589   else
4590   {
4591     BlockChainStream *stream;
4592 
4593     stream = *StorageImpl_GetCachedBlockChainStream(This, index);
4594     if (!stream) return E_OUTOFMEMORY;
4595 
4596     return BlockChainStream_WriteAt(stream, offset, size, buffer, bytesWritten);
4597   }
4598 }
4599 
4600 static HRESULT StorageImpl_StreamLink(StorageBaseImpl *base, DirRef dst,
4601   DirRef src)
4602 {
4603   StorageImpl *This = (StorageImpl*)base;
4604   DirEntry dst_data, src_data;
4605   HRESULT hr;
4606 
4607   hr = StorageImpl_ReadDirEntry(This, dst, &dst_data);
4608 
4609   if (SUCCEEDED(hr))
4610     hr = StorageImpl_ReadDirEntry(This, src, &src_data);
4611 
4612   if (SUCCEEDED(hr))
4613   {
4614     StorageImpl_DeleteCachedBlockChainStream(This, src);
4615     dst_data.startingBlock = src_data.startingBlock;
4616     dst_data.size = src_data.size;
4617 
4618     hr = StorageImpl_WriteDirEntry(This, dst, &dst_data);
4619   }
4620 
4621   return hr;
4622 }
4623 
4624 static HRESULT StorageImpl_Refresh(StorageImpl *This, BOOL new_object, BOOL create)
4625 {
4626   HRESULT hr=S_OK;
4627   DirEntry currentEntry;
4628   DirRef      currentEntryRef;
4629   BlockChainStream *blockChainStream;
4630 
4631   if (create)
4632   {
4633     ULARGE_INTEGER size;
4634     BYTE bigBlockBuffer[MAX_BIG_BLOCK_SIZE];
4635 
4636     /* Discard any existing data. */
4637     size.QuadPart = 0;
4638     ILockBytes_SetSize(This->lockBytes, size);
4639 
4640     /*
4641      * Initialize all header variables:
4642      * - The big block depot consists of one block and it is at block 0
4643      * - The directory table starts at block 1
4644      * - There is no small block depot
4645      */
4646     memset( This->bigBlockDepotStart,
4647             BLOCK_UNUSED,
4648             sizeof(This->bigBlockDepotStart));
4649 
4650     This->bigBlockDepotCount    = 1;
4651     This->bigBlockDepotStart[0] = 0;
4652     This->rootStartBlock        = 1;
4653     This->smallBlockLimit       = LIMIT_TO_USE_SMALL_BLOCK;
4654     This->smallBlockDepotStart  = BLOCK_END_OF_CHAIN;
4655     if (This->bigBlockSize == 4096)
4656       This->bigBlockSizeBits    = MAX_BIG_BLOCK_SIZE_BITS;
4657     else
4658       This->bigBlockSizeBits    = MIN_BIG_BLOCK_SIZE_BITS;
4659     This->smallBlockSizeBits    = DEF_SMALL_BLOCK_SIZE_BITS;
4660     This->extBigBlockDepotStart = BLOCK_END_OF_CHAIN;
4661     This->extBigBlockDepotCount = 0;
4662 
4663     StorageImpl_SaveFileHeader(This);
4664 
4665     /*
4666      * Add one block for the big block depot and one block for the directory table
4667      */
4668     size.u.HighPart = 0;
4669     size.u.LowPart  = This->bigBlockSize * 3;
4670     ILockBytes_SetSize(This->lockBytes, size);
4671 
4672     /*
4673      * Initialize the big block depot
4674      */
4675     memset(bigBlockBuffer, BLOCK_UNUSED, This->bigBlockSize);
4676     StorageUtl_WriteDWord(bigBlockBuffer, 0, BLOCK_SPECIAL);
4677     StorageUtl_WriteDWord(bigBlockBuffer, sizeof(ULONG), BLOCK_END_OF_CHAIN);
4678     StorageImpl_WriteBigBlock(This, 0, bigBlockBuffer);
4679   }
4680   else
4681   {
4682     /*
4683      * Load the header for the file.
4684      */
4685     hr = StorageImpl_LoadFileHeader(This);
4686 
4687     if (FAILED(hr))
4688     {
4689       return hr;
4690     }
4691   }
4692 
4693   /*
4694    * There is no block depot cached yet.
4695    */
4696   This->indexBlockDepotCached = 0xFFFFFFFF;
4697   This->indexExtBlockDepotCached = 0xFFFFFFFF;
4698 
4699   /*
4700    * Start searching for free blocks with block 0.
4701    */
4702   This->prevFreeBlock = 0;
4703 
4704   This->firstFreeSmallBlock = 0;
4705 
4706   /* Read the extended big block depot locations. */
4707   if (This->extBigBlockDepotCount != 0)
4708   {
4709     ULONG current_block = This->extBigBlockDepotStart;
4710     ULONG cache_size = This->extBigBlockDepotCount * 2;
4711     ULONG i;
4712 
4713     This->extBigBlockDepotLocations = HeapAlloc(GetProcessHeap(), 0, sizeof(ULONG) * cache_size);
4714     if (!This->extBigBlockDepotLocations)
4715     {
4716       return E_OUTOFMEMORY;
4717     }
4718 
4719     This->extBigBlockDepotLocationsSize = cache_size;
4720 
4721     for (i=0; i<This->extBigBlockDepotCount; i++)
4722     {
4723       if (current_block == BLOCK_END_OF_CHAIN)
4724       {
4725         WARN("File has too few extended big block depot blocks.\n");
4726         return STG_E_DOCFILECORRUPT;
4727       }
4728       This->extBigBlockDepotLocations[i] = current_block;
4729       current_block = Storage32Impl_GetNextExtendedBlock(This, current_block);
4730     }
4731   }
4732   else
4733   {
4734     This->extBigBlockDepotLocations = NULL;
4735     This->extBigBlockDepotLocationsSize = 0;
4736   }
4737 
4738   /*
4739    * Create the block chain abstractions.
4740    */
4741   if(!(blockChainStream =
4742        BlockChainStream_Construct(This, &This->rootStartBlock, DIRENTRY_NULL)))
4743   {
4744     return STG_E_READFAULT;
4745   }
4746   if (!new_object)
4747     BlockChainStream_Destroy(This->rootBlockChain);
4748   This->rootBlockChain = blockChainStream;
4749 
4750   if(!(blockChainStream =
4751        BlockChainStream_Construct(This, &This->smallBlockDepotStart,
4752 				  DIRENTRY_NULL)))
4753   {
4754     return STG_E_READFAULT;
4755   }
4756   if (!new_object)
4757     BlockChainStream_Destroy(This->smallBlockDepotChain);
4758   This->smallBlockDepotChain = blockChainStream;
4759 
4760   /*
4761    * Write the root storage entry (memory only)
4762    */
4763   if (create)
4764   {
4765     static const WCHAR rootentryW[] = {'R','o','o','t',' ','E','n','t','r','y',0};
4766     DirEntry rootEntry;
4767     /*
4768      * Initialize the directory table
4769      */
4770     memset(&rootEntry, 0, sizeof(rootEntry));
4771     strcpyW(rootEntry.name, rootentryW);
4772     rootEntry.sizeOfNameString = sizeof(rootentryW);
4773     rootEntry.stgType          = STGTY_ROOT;
4774     rootEntry.leftChild        = DIRENTRY_NULL;
4775     rootEntry.rightChild       = DIRENTRY_NULL;
4776     rootEntry.dirRootEntry     = DIRENTRY_NULL;
4777     rootEntry.startingBlock    = BLOCK_END_OF_CHAIN;
4778     rootEntry.size.u.HighPart  = 0;
4779     rootEntry.size.u.LowPart   = 0;
4780 
4781     StorageImpl_WriteDirEntry(This, 0, &rootEntry);
4782   }
4783 
4784   /*
4785    * Find the ID of the root storage.
4786    */
4787   currentEntryRef = 0;
4788 
4789   do
4790   {
4791     hr = StorageImpl_ReadDirEntry(
4792                       This,
4793                       currentEntryRef,
4794                       &currentEntry);
4795 
4796     if (SUCCEEDED(hr))
4797     {
4798       if ( (currentEntry.sizeOfNameString != 0 ) &&
4799            (currentEntry.stgType          == STGTY_ROOT) )
4800       {
4801         This->base.storageDirEntry = currentEntryRef;
4802       }
4803     }
4804 
4805     currentEntryRef++;
4806 
4807   } while (SUCCEEDED(hr) && (This->base.storageDirEntry == DIRENTRY_NULL) );
4808 
4809   if (FAILED(hr))
4810   {
4811     return STG_E_READFAULT;
4812   }
4813 
4814   /*
4815    * Create the block chain abstraction for the small block root chain.
4816    */
4817   if(!(blockChainStream =
4818        BlockChainStream_Construct(This, NULL, This->base.storageDirEntry)))
4819   {
4820     return STG_E_READFAULT;
4821   }
4822   if (!new_object)
4823     BlockChainStream_Destroy(This->smallBlockRootChain);
4824   This->smallBlockRootChain = blockChainStream;
4825 
4826   if (!new_object)
4827   {
4828     int i;
4829     for (i=0; i<BLOCKCHAIN_CACHE_SIZE; i++)
4830     {
4831       BlockChainStream_Destroy(This->blockChainCache[i]);
4832       This->blockChainCache[i] = NULL;
4833     }
4834   }
4835 
4836   return hr;
4837 }
4838 
4839 static HRESULT StorageImpl_GetTransactionSig(StorageBaseImpl *base,
4840   ULONG* result, BOOL refresh)
4841 {
4842   StorageImpl *This = (StorageImpl*)base;
4843   HRESULT hr=S_OK;
4844   DWORD oldTransactionSig = This->transactionSig;
4845 
4846   if (refresh)
4847   {
4848     ULARGE_INTEGER offset;
4849     ULONG bytes_read;
4850     BYTE data[4];
4851 
4852     offset.u.HighPart = 0;
4853     offset.u.LowPart = OFFSET_TRANSACTIONSIG;
4854     hr = StorageImpl_ReadAt(This, offset, data, 4, &bytes_read);
4855 
4856     if (SUCCEEDED(hr))
4857     {
4858       StorageUtl_ReadDWord(data, 0, &This->transactionSig);
4859 
4860       if (oldTransactionSig != This->transactionSig)
4861       {
4862         /* Someone else wrote to this, so toss all cached information. */
4863         TRACE("signature changed\n");
4864 
4865         hr = StorageImpl_Refresh(This, FALSE, FALSE);
4866       }
4867 
4868       if (FAILED(hr))
4869         This->transactionSig = oldTransactionSig;
4870     }
4871   }
4872 
4873   *result = This->transactionSig;
4874 
4875   return hr;
4876 }
4877 
4878 static HRESULT StorageImpl_SetTransactionSig(StorageBaseImpl *base,
4879   ULONG value)
4880 {
4881   StorageImpl *This = (StorageImpl*)base;
4882 
4883   This->transactionSig = value;
4884   StorageImpl_SaveFileHeader(This);
4885 
4886   return S_OK;
4887 }
4888 
4889 static HRESULT StorageImpl_LockRegion(StorageImpl *This, ULARGE_INTEGER offset,
4890     ULARGE_INTEGER cb, DWORD dwLockType, BOOL *supported)
4891 {
4892     if ((dwLockType & This->locks_supported) == 0)
4893     {
4894         if (supported) *supported = FALSE;
4895         return S_OK;
4896     }
4897 
4898     if (supported) *supported = TRUE;
4899     return ILockBytes_LockRegion(This->lockBytes, offset, cb, dwLockType);
4900 }
4901 
4902 static HRESULT StorageImpl_UnlockRegion(StorageImpl *This, ULARGE_INTEGER offset,
4903     ULARGE_INTEGER cb, DWORD dwLockType)
4904 {
4905     if ((dwLockType & This->locks_supported) == 0)
4906         return S_OK;
4907 
4908     return ILockBytes_UnlockRegion(This->lockBytes, offset, cb, dwLockType);
4909 }
4910 
4911 /* Internal function */
4912 static HRESULT StorageImpl_LockRegionSync(StorageImpl *This, ULARGE_INTEGER offset,
4913     ULARGE_INTEGER cb, DWORD dwLockType, BOOL *supported)
4914 {
4915     HRESULT hr;
4916     int delay = 0;
4917     DWORD start_time = GetTickCount();
4918     DWORD last_sanity_check = start_time;
4919     ULARGE_INTEGER sanity_offset, sanity_cb;
4920 
4921     sanity_offset.QuadPart = RANGELOCK_UNK1_FIRST;
4922     sanity_cb.QuadPart = RANGELOCK_UNK1_LAST - RANGELOCK_UNK1_FIRST + 1;
4923 
4924     do
4925     {
4926         hr = StorageImpl_LockRegion(This, offset, cb, dwLockType, supported);
4927 
4928         if (hr == STG_E_ACCESSDENIED || hr == STG_E_LOCKVIOLATION)
4929         {
4930             DWORD current_time = GetTickCount();
4931             if (current_time - start_time >= 20000)
4932             {
4933                 /* timeout */
4934                 break;
4935             }
4936             if (current_time - last_sanity_check >= 500)
4937             {
4938                 /* Any storage implementation with the file open in a
4939                  * shared mode should not lock these bytes for writing. However,
4940                  * some programs (LibreOffice Writer) will keep ALL bytes locked
4941                  * when opening in exclusive mode. We can use a read lock to
4942                  * detect this case early, and not hang a full 20 seconds.
4943                  *
4944                  * This can collide with another attempt to open the file in
4945                  * exclusive mode, but it's unlikely, and someone would fail anyway. */
4946                 hr = StorageImpl_LockRegion(This, sanity_offset, sanity_cb, WINE_LOCK_READ, NULL);
4947                 if (hr == STG_E_ACCESSDENIED || hr == STG_E_LOCKVIOLATION)
4948                     break;
4949                 if (SUCCEEDED(hr))
4950                 {
4951                     StorageImpl_UnlockRegion(This, sanity_offset, sanity_cb, WINE_LOCK_READ);
4952                     hr = STG_E_ACCESSDENIED;
4953                 }
4954 
4955                 last_sanity_check = current_time;
4956             }
4957             Sleep(delay);
4958             if (delay < 150) delay++;
4959         }
4960     } while (hr == STG_E_ACCESSDENIED || hr == STG_E_LOCKVIOLATION);
4961 
4962     return hr;
4963 }
4964 
4965 static HRESULT StorageImpl_LockTransaction(StorageBaseImpl *base, BOOL write)
4966 {
4967   StorageImpl *This = (StorageImpl*)base;
4968   HRESULT hr;
4969   ULARGE_INTEGER offset, cb;
4970 
4971   if (write)
4972   {
4973     /* Synchronous grab of second priority range, the commit lock, and the
4974      * lock-checking lock. */
4975     offset.QuadPart = RANGELOCK_TRANSACTION_FIRST;
4976     cb.QuadPart = RANGELOCK_TRANSACTION_LAST - RANGELOCK_TRANSACTION_FIRST + 1;
4977   }
4978   else
4979   {
4980     offset.QuadPart = RANGELOCK_COMMIT;
4981     cb.QuadPart = 1;
4982   }
4983 
4984   hr = StorageImpl_LockRegionSync(This, offset, cb, LOCK_ONLYONCE, NULL);
4985 
4986   return hr;
4987 }
4988 
4989 static HRESULT StorageImpl_UnlockTransaction(StorageBaseImpl *base, BOOL write)
4990 {
4991   StorageImpl *This = (StorageImpl*)base;
4992   HRESULT hr;
4993   ULARGE_INTEGER offset, cb;
4994 
4995   if (write)
4996   {
4997     offset.QuadPart = RANGELOCK_TRANSACTION_FIRST;
4998     cb.QuadPart = RANGELOCK_TRANSACTION_LAST - RANGELOCK_TRANSACTION_FIRST + 1;
4999   }
5000   else
5001   {
5002     offset.QuadPart = RANGELOCK_COMMIT;
5003     cb.QuadPart = 1;
5004   }
5005 
5006   hr = StorageImpl_UnlockRegion(This, offset, cb, LOCK_ONLYONCE);
5007 
5008   return hr;
5009 }
5010 
5011 static HRESULT StorageImpl_GetFilename(StorageBaseImpl* iface, LPWSTR *result)
5012 {
5013   StorageImpl *This = (StorageImpl*) iface;
5014   STATSTG statstg;
5015   HRESULT hr;
5016 
5017   hr = ILockBytes_Stat(This->lockBytes, &statstg, 0);
5018 
5019   *result = statstg.pwcsName;
5020 
5021   return hr;
5022 }
5023 
5024 static HRESULT StorageImpl_CheckLockRange(StorageImpl *This, ULONG start,
5025     ULONG end, HRESULT fail_hr)
5026 {
5027     HRESULT hr;
5028     ULARGE_INTEGER offset, cb;
5029 
5030     offset.QuadPart = start;
5031     cb.QuadPart = 1 + end - start;
5032 
5033     hr = StorageImpl_LockRegion(This, offset, cb, LOCK_ONLYONCE, NULL);
5034     if (SUCCEEDED(hr)) StorageImpl_UnlockRegion(This, offset, cb, LOCK_ONLYONCE);
5035 
5036     if (FAILED(hr))
5037         return fail_hr;
5038     else
5039         return S_OK;
5040 }
5041 
5042 static HRESULT StorageImpl_LockOne(StorageImpl *This, ULONG start, ULONG end)
5043 {
5044     HRESULT hr=S_OK;
5045     int i, j;
5046     ULARGE_INTEGER offset, cb;
5047 
5048     cb.QuadPart = 1;
5049 
5050     for (i=start; i<=end; i++)
5051     {
5052         offset.QuadPart = i;
5053         hr = StorageImpl_LockRegion(This, offset, cb, LOCK_ONLYONCE, NULL);
5054         if (hr != STG_E_ACCESSDENIED && hr != STG_E_LOCKVIOLATION)
5055             break;
5056     }
5057 
5058     if (SUCCEEDED(hr))
5059     {
5060         for (j = 0; j < ARRAY_SIZE(This->locked_bytes); j++)
5061         {
5062             if (This->locked_bytes[j] == 0)
5063             {
5064                 This->locked_bytes[j] = i;
5065                 break;
5066             }
5067         }
5068     }
5069 
5070     return hr;
5071 }
5072 
5073 static HRESULT StorageImpl_GrabLocks(StorageImpl *This, DWORD openFlags)
5074 {
5075     HRESULT hr;
5076     ULARGE_INTEGER offset;
5077     ULARGE_INTEGER cb;
5078     DWORD share_mode = STGM_SHARE_MODE(openFlags);
5079     BOOL supported;
5080 
5081     if (openFlags & STGM_NOSNAPSHOT)
5082     {
5083         /* STGM_NOSNAPSHOT implies deny write */
5084         if (share_mode == STGM_SHARE_DENY_READ) share_mode = STGM_SHARE_EXCLUSIVE;
5085         else if (share_mode != STGM_SHARE_EXCLUSIVE) share_mode = STGM_SHARE_DENY_WRITE;
5086     }
5087 
5088     /* Wrap all other locking inside a single lock so we can check ranges safely */
5089     offset.QuadPart = RANGELOCK_CHECKLOCKS;
5090     cb.QuadPart = 1;
5091     hr = StorageImpl_LockRegionSync(This, offset, cb, LOCK_ONLYONCE, &supported);
5092 
5093     /* If the ILockBytes doesn't support locking that's ok. */
5094     if (!supported) return S_OK;
5095     else if (FAILED(hr)) return hr;
5096 
5097     hr = S_OK;
5098 
5099     /* First check for any conflicting locks. */
5100     if ((openFlags & STGM_PRIORITY) == STGM_PRIORITY)
5101         hr = StorageImpl_CheckLockRange(This, RANGELOCK_COMMIT, RANGELOCK_COMMIT, STG_E_LOCKVIOLATION);
5102 
5103     if (SUCCEEDED(hr) && (STGM_ACCESS_MODE(openFlags) != STGM_WRITE))
5104         hr = StorageImpl_CheckLockRange(This, RANGELOCK_DENY_READ_FIRST, RANGELOCK_DENY_READ_LAST, STG_E_SHAREVIOLATION);
5105 
5106     if (SUCCEEDED(hr) && (STGM_ACCESS_MODE(openFlags) != STGM_READ))
5107         hr = StorageImpl_CheckLockRange(This, RANGELOCK_DENY_WRITE_FIRST, RANGELOCK_DENY_WRITE_LAST, STG_E_SHAREVIOLATION);
5108 
5109     if (SUCCEEDED(hr) && (share_mode == STGM_SHARE_DENY_READ || share_mode == STGM_SHARE_EXCLUSIVE))
5110         hr = StorageImpl_CheckLockRange(This, RANGELOCK_READ_FIRST, RANGELOCK_READ_LAST, STG_E_LOCKVIOLATION);
5111 
5112     if (SUCCEEDED(hr) && (share_mode == STGM_SHARE_DENY_WRITE || share_mode == STGM_SHARE_EXCLUSIVE))
5113         hr = StorageImpl_CheckLockRange(This, RANGELOCK_WRITE_FIRST, RANGELOCK_WRITE_LAST, STG_E_LOCKVIOLATION);
5114 
5115     if (SUCCEEDED(hr) && STGM_ACCESS_MODE(openFlags) == STGM_READ && share_mode == STGM_SHARE_EXCLUSIVE)
5116     {
5117         hr = StorageImpl_CheckLockRange(This, 0, RANGELOCK_CHECKLOCKS-1, STG_E_LOCKVIOLATION);
5118 
5119         if (SUCCEEDED(hr))
5120             hr = StorageImpl_CheckLockRange(This, RANGELOCK_CHECKLOCKS+1, RANGELOCK_LAST, STG_E_LOCKVIOLATION);
5121     }
5122 
5123     /* Then grab our locks. */
5124     if (SUCCEEDED(hr) && (openFlags & STGM_PRIORITY) == STGM_PRIORITY)
5125     {
5126         hr = StorageImpl_LockOne(This, RANGELOCK_PRIORITY1_FIRST, RANGELOCK_PRIORITY1_LAST);
5127         if (SUCCEEDED(hr))
5128             hr = StorageImpl_LockOne(This, RANGELOCK_PRIORITY2_FIRST, RANGELOCK_PRIORITY2_LAST);
5129     }
5130 
5131     if (SUCCEEDED(hr) && (STGM_ACCESS_MODE(openFlags) != STGM_WRITE))
5132         hr = StorageImpl_LockOne(This, RANGELOCK_READ_FIRST, RANGELOCK_READ_LAST);
5133 
5134     if (SUCCEEDED(hr) && (STGM_ACCESS_MODE(openFlags) != STGM_READ))
5135         hr = StorageImpl_LockOne(This, RANGELOCK_WRITE_FIRST, RANGELOCK_WRITE_LAST);
5136 
5137     if (SUCCEEDED(hr) && (share_mode == STGM_SHARE_DENY_READ || share_mode == STGM_SHARE_EXCLUSIVE))
5138         hr = StorageImpl_LockOne(This, RANGELOCK_DENY_READ_FIRST, RANGELOCK_DENY_READ_LAST);
5139 
5140     if (SUCCEEDED(hr) && (share_mode == STGM_SHARE_DENY_WRITE || share_mode == STGM_SHARE_EXCLUSIVE))
5141         hr = StorageImpl_LockOne(This, RANGELOCK_DENY_WRITE_FIRST, RANGELOCK_DENY_WRITE_LAST);
5142 
5143     if (SUCCEEDED(hr) && (openFlags & STGM_NOSNAPSHOT) == STGM_NOSNAPSHOT)
5144         hr = StorageImpl_LockOne(This, RANGELOCK_NOSNAPSHOT_FIRST, RANGELOCK_NOSNAPSHOT_LAST);
5145 
5146     offset.QuadPart = RANGELOCK_CHECKLOCKS;
5147     cb.QuadPart = 1;
5148     StorageImpl_UnlockRegion(This, offset, cb, LOCK_ONLYONCE);
5149 
5150     return hr;
5151 }
5152 
5153 static HRESULT StorageImpl_Flush(StorageBaseImpl *storage)
5154 {
5155   StorageImpl *This = (StorageImpl*)storage;
5156   int i;
5157   HRESULT hr;
5158   TRACE("(%p)\n", This);
5159 
5160   hr = BlockChainStream_Flush(This->smallBlockRootChain);
5161 
5162   if (SUCCEEDED(hr))
5163     hr = BlockChainStream_Flush(This->rootBlockChain);
5164 
5165   if (SUCCEEDED(hr))
5166     hr = BlockChainStream_Flush(This->smallBlockDepotChain);
5167 
5168   for (i=0; SUCCEEDED(hr) && i<BLOCKCHAIN_CACHE_SIZE; i++)
5169     if (This->blockChainCache[i])
5170       hr = BlockChainStream_Flush(This->blockChainCache[i]);
5171 
5172   if (SUCCEEDED(hr))
5173     hr = ILockBytes_Flush(This->lockBytes);
5174 
5175   return hr;
5176 }
5177 
5178 static void StorageImpl_Invalidate(StorageBaseImpl* iface)
5179 {
5180   StorageImpl *This = (StorageImpl*) iface;
5181 
5182   StorageBaseImpl_DeleteAll(&This->base);
5183 
5184   This->base.reverted = TRUE;
5185 }
5186 
5187 static void StorageImpl_Destroy(StorageBaseImpl* iface)
5188 {
5189   StorageImpl *This = (StorageImpl*) iface;
5190   int i;
5191   TRACE("(%p)\n", This);
5192 
5193   StorageImpl_Flush(iface);
5194 
5195   StorageImpl_Invalidate(iface);
5196 
5197   HeapFree(GetProcessHeap(), 0, This->extBigBlockDepotLocations);
5198 
5199   BlockChainStream_Destroy(This->smallBlockRootChain);
5200   BlockChainStream_Destroy(This->rootBlockChain);
5201   BlockChainStream_Destroy(This->smallBlockDepotChain);
5202 
5203   for (i = 0; i < BLOCKCHAIN_CACHE_SIZE; i++)
5204     BlockChainStream_Destroy(This->blockChainCache[i]);
5205 
5206   for (i = 0; i < ARRAY_SIZE(This->locked_bytes); i++)
5207   {
5208     ULARGE_INTEGER offset, cb;
5209     cb.QuadPart = 1;
5210     if (This->locked_bytes[i] != 0)
5211     {
5212       offset.QuadPart = This->locked_bytes[i];
5213       StorageImpl_UnlockRegion(This, offset, cb, LOCK_ONLYONCE);
5214     }
5215   }
5216 
5217   if (This->lockBytes)
5218     ILockBytes_Release(This->lockBytes);
5219   HeapFree(GetProcessHeap(), 0, This);
5220 }
5221 
5222 
5223 static const StorageBaseImplVtbl StorageImpl_BaseVtbl =
5224 {
5225   StorageImpl_Destroy,
5226   StorageImpl_Invalidate,
5227   StorageImpl_Flush,
5228   StorageImpl_GetFilename,
5229   StorageImpl_CreateDirEntry,
5230   StorageImpl_BaseWriteDirEntry,
5231   StorageImpl_BaseReadDirEntry,
5232   StorageImpl_DestroyDirEntry,
5233   StorageImpl_StreamReadAt,
5234   StorageImpl_StreamWriteAt,
5235   StorageImpl_StreamSetSize,
5236   StorageImpl_StreamLink,
5237   StorageImpl_GetTransactionSig,
5238   StorageImpl_SetTransactionSig,
5239   StorageImpl_LockTransaction,
5240   StorageImpl_UnlockTransaction
5241 };
5242 
5243 
5244 /*
5245  * Virtual function table for the IStorageBaseImpl class.
5246  */
5247 static const IStorageVtbl StorageImpl_Vtbl =
5248 {
5249     StorageBaseImpl_QueryInterface,
5250     StorageBaseImpl_AddRef,
5251     StorageBaseImpl_Release,
5252     StorageBaseImpl_CreateStream,
5253     StorageBaseImpl_OpenStream,
5254     StorageBaseImpl_CreateStorage,
5255     StorageBaseImpl_OpenStorage,
5256     StorageBaseImpl_CopyTo,
5257     StorageBaseImpl_MoveElementTo,
5258     StorageBaseImpl_Commit,
5259     StorageBaseImpl_Revert,
5260     StorageBaseImpl_EnumElements,
5261     StorageBaseImpl_DestroyElement,
5262     StorageBaseImpl_RenameElement,
5263     StorageBaseImpl_SetElementTimes,
5264     StorageBaseImpl_SetClass,
5265     StorageBaseImpl_SetStateBits,
5266     StorageBaseImpl_Stat
5267 };
5268 
5269 static HRESULT StorageImpl_Construct(
5270   HANDLE       hFile,
5271   LPCOLESTR    pwcsName,
5272   ILockBytes*  pLkbyt,
5273   DWORD        openFlags,
5274   BOOL         fileBased,
5275   BOOL         create,
5276   ULONG        sector_size,
5277   StorageImpl** result)
5278 {
5279   StorageImpl* This;
5280   HRESULT hr = S_OK;
5281   STATSTG stat;
5282 
5283   if ( FAILED( validateSTGM(openFlags) ))
5284     return STG_E_INVALIDFLAG;
5285 
5286   This = HeapAlloc(GetProcessHeap(), 0, sizeof(StorageImpl));
5287   if (!This)
5288     return E_OUTOFMEMORY;
5289 
5290   memset(This, 0, sizeof(StorageImpl));
5291 
5292   list_init(&This->base.strmHead);
5293 
5294   list_init(&This->base.storageHead);
5295 
5296   This->base.IStorage_iface.lpVtbl = &StorageImpl_Vtbl;
5297   This->base.IPropertySetStorage_iface.lpVtbl = &IPropertySetStorage_Vtbl;
5298   This->base.IDirectWriterLock_iface.lpVtbl = &DirectWriterLockVtbl;
5299   This->base.baseVtbl = &StorageImpl_BaseVtbl;
5300   This->base.openFlags = (openFlags & ~STGM_CREATE);
5301   This->base.ref = 1;
5302   This->base.create = create;
5303 
5304   if (openFlags == (STGM_DIRECT_SWMR|STGM_READWRITE|STGM_SHARE_DENY_WRITE))
5305     This->base.lockingrole = SWMR_Writer;
5306   else if (openFlags == (STGM_DIRECT_SWMR|STGM_READ|STGM_SHARE_DENY_NONE))
5307     This->base.lockingrole = SWMR_Reader;
5308   else
5309     This->base.lockingrole = SWMR_None;
5310 
5311   This->base.reverted = FALSE;
5312 
5313   /*
5314    * Initialize the big block cache.
5315    */
5316   This->bigBlockSize   = sector_size;
5317   This->smallBlockSize = DEF_SMALL_BLOCK_SIZE;
5318   if (hFile)
5319     hr = FileLockBytesImpl_Construct(hFile, openFlags, pwcsName, &This->lockBytes);
5320   else
5321   {
5322     This->lockBytes = pLkbyt;
5323     ILockBytes_AddRef(pLkbyt);
5324   }
5325 
5326   if (SUCCEEDED(hr))
5327     hr = ILockBytes_Stat(This->lockBytes, &stat, STATFLAG_NONAME);
5328 
5329   if (SUCCEEDED(hr))
5330   {
5331     This->locks_supported = stat.grfLocksSupported;
5332     if (!hFile)
5333         /* Don't try to use wine-internal locking flag with custom ILockBytes */
5334         This->locks_supported &= ~WINE_LOCK_READ;
5335 
5336     hr = StorageImpl_GrabLocks(This, openFlags);
5337   }
5338 
5339   if (SUCCEEDED(hr))
5340     hr = StorageImpl_Refresh(This, TRUE, create);
5341 
5342   if (FAILED(hr))
5343   {
5344     IStorage_Release(&This->base.IStorage_iface);
5345     *result = NULL;
5346   }
5347   else
5348   {
5349     StorageImpl_Flush(&This->base);
5350     *result = This;
5351   }
5352 
5353   return hr;
5354 }
5355 
5356 
5357 /************************************************************************
5358  * StorageInternalImpl implementation
5359  ***********************************************************************/
5360 
5361 static void StorageInternalImpl_Invalidate( StorageBaseImpl *base )
5362 {
5363   StorageInternalImpl* This = (StorageInternalImpl*) base;
5364 
5365   if (!This->base.reverted)
5366   {
5367     TRACE("Storage invalidated (stg=%p)\n", This);
5368 
5369     This->base.reverted = TRUE;
5370 
5371     This->parentStorage = NULL;
5372 
5373     StorageBaseImpl_DeleteAll(&This->base);
5374 
5375     list_remove(&This->ParentListEntry);
5376   }
5377 }
5378 
5379 static void StorageInternalImpl_Destroy( StorageBaseImpl *iface)
5380 {
5381   StorageInternalImpl* This = (StorageInternalImpl*) iface;
5382 
5383   StorageInternalImpl_Invalidate(&This->base);
5384 
5385   HeapFree(GetProcessHeap(), 0, This);
5386 }
5387 
5388 static HRESULT StorageInternalImpl_Flush(StorageBaseImpl* iface)
5389 {
5390   StorageInternalImpl* This = (StorageInternalImpl*) iface;
5391 
5392   return StorageBaseImpl_Flush(This->parentStorage);
5393 }
5394 
5395 static HRESULT StorageInternalImpl_GetFilename(StorageBaseImpl* iface, LPWSTR *result)
5396 {
5397   StorageInternalImpl* This = (StorageInternalImpl*) iface;
5398 
5399   return StorageBaseImpl_GetFilename(This->parentStorage, result);
5400 }
5401 
5402 static HRESULT StorageInternalImpl_CreateDirEntry(StorageBaseImpl *base,
5403   const DirEntry *newData, DirRef *index)
5404 {
5405   StorageInternalImpl* This = (StorageInternalImpl*) base;
5406 
5407   return StorageBaseImpl_CreateDirEntry(This->parentStorage,
5408     newData, index);
5409 }
5410 
5411 static HRESULT StorageInternalImpl_WriteDirEntry(StorageBaseImpl *base,
5412   DirRef index, const DirEntry *data)
5413 {
5414   StorageInternalImpl* This = (StorageInternalImpl*) base;
5415 
5416   return StorageBaseImpl_WriteDirEntry(This->parentStorage,
5417     index, data);
5418 }
5419 
5420 static HRESULT StorageInternalImpl_ReadDirEntry(StorageBaseImpl *base,
5421   DirRef index, DirEntry *data)
5422 {
5423   StorageInternalImpl* This = (StorageInternalImpl*) base;
5424 
5425   return StorageBaseImpl_ReadDirEntry(This->parentStorage,
5426     index, data);
5427 }
5428 
5429 static HRESULT StorageInternalImpl_DestroyDirEntry(StorageBaseImpl *base,
5430   DirRef index)
5431 {
5432   StorageInternalImpl* This = (StorageInternalImpl*) base;
5433 
5434   return StorageBaseImpl_DestroyDirEntry(This->parentStorage,
5435     index);
5436 }
5437 
5438 static HRESULT StorageInternalImpl_StreamReadAt(StorageBaseImpl *base,
5439   DirRef index, ULARGE_INTEGER offset, ULONG size, void *buffer, ULONG *bytesRead)
5440 {
5441   StorageInternalImpl* This = (StorageInternalImpl*) base;
5442 
5443   return StorageBaseImpl_StreamReadAt(This->parentStorage,
5444     index, offset, size, buffer, bytesRead);
5445 }
5446 
5447 static HRESULT StorageInternalImpl_StreamWriteAt(StorageBaseImpl *base,
5448   DirRef index, ULARGE_INTEGER offset, ULONG size, const void *buffer, ULONG *bytesWritten)
5449 {
5450   StorageInternalImpl* This = (StorageInternalImpl*) base;
5451 
5452   return StorageBaseImpl_StreamWriteAt(This->parentStorage,
5453     index, offset, size, buffer, bytesWritten);
5454 }
5455 
5456 static HRESULT StorageInternalImpl_StreamSetSize(StorageBaseImpl *base,
5457   DirRef index, ULARGE_INTEGER newsize)
5458 {
5459   StorageInternalImpl* This = (StorageInternalImpl*) base;
5460 
5461   return StorageBaseImpl_StreamSetSize(This->parentStorage,
5462     index, newsize);
5463 }
5464 
5465 static HRESULT StorageInternalImpl_StreamLink(StorageBaseImpl *base,
5466   DirRef dst, DirRef src)
5467 {
5468   StorageInternalImpl* This = (StorageInternalImpl*) base;
5469 
5470   return StorageBaseImpl_StreamLink(This->parentStorage,
5471     dst, src);
5472 }
5473 
5474 static HRESULT StorageInternalImpl_GetTransactionSig(StorageBaseImpl *base,
5475   ULONG* result, BOOL refresh)
5476 {
5477   return E_NOTIMPL;
5478 }
5479 
5480 static HRESULT StorageInternalImpl_SetTransactionSig(StorageBaseImpl *base,
5481   ULONG value)
5482 {
5483   return E_NOTIMPL;
5484 }
5485 
5486 static HRESULT StorageInternalImpl_LockTransaction(StorageBaseImpl *base, BOOL write)
5487 {
5488   return E_NOTIMPL;
5489 }
5490 
5491 static HRESULT StorageInternalImpl_UnlockTransaction(StorageBaseImpl *base, BOOL write)
5492 {
5493   return E_NOTIMPL;
5494 }
5495 
5496 /******************************************************************************
5497 **
5498 ** StorageInternalImpl_Commit
5499 **
5500 */
5501 static HRESULT WINAPI StorageInternalImpl_Commit(
5502   IStorage*            iface,
5503   DWORD                  grfCommitFlags)  /* [in] */
5504 {
5505   StorageBaseImpl* This = impl_from_IStorage(iface);
5506   TRACE("(%p,%x)\n", iface, grfCommitFlags);
5507   return StorageBaseImpl_Flush(This);
5508 }
5509 
5510 /******************************************************************************
5511 **
5512 ** StorageInternalImpl_Revert
5513 **
5514 */
5515 static HRESULT WINAPI StorageInternalImpl_Revert(
5516   IStorage*            iface)
5517 {
5518   FIXME("(%p): stub\n", iface);
5519   return S_OK;
5520 }
5521 
5522 /*
5523  * Virtual function table for the StorageInternalImpl class.
5524  */
5525 static const IStorageVtbl StorageInternalImpl_Vtbl =
5526 {
5527     StorageBaseImpl_QueryInterface,
5528     StorageBaseImpl_AddRef,
5529     StorageBaseImpl_Release,
5530     StorageBaseImpl_CreateStream,
5531     StorageBaseImpl_OpenStream,
5532     StorageBaseImpl_CreateStorage,
5533     StorageBaseImpl_OpenStorage,
5534     StorageBaseImpl_CopyTo,
5535     StorageBaseImpl_MoveElementTo,
5536     StorageInternalImpl_Commit,
5537     StorageInternalImpl_Revert,
5538     StorageBaseImpl_EnumElements,
5539     StorageBaseImpl_DestroyElement,
5540     StorageBaseImpl_RenameElement,
5541     StorageBaseImpl_SetElementTimes,
5542     StorageBaseImpl_SetClass,
5543     StorageBaseImpl_SetStateBits,
5544     StorageBaseImpl_Stat
5545 };
5546 
5547 static const StorageBaseImplVtbl StorageInternalImpl_BaseVtbl =
5548 {
5549   StorageInternalImpl_Destroy,
5550   StorageInternalImpl_Invalidate,
5551   StorageInternalImpl_Flush,
5552   StorageInternalImpl_GetFilename,
5553   StorageInternalImpl_CreateDirEntry,
5554   StorageInternalImpl_WriteDirEntry,
5555   StorageInternalImpl_ReadDirEntry,
5556   StorageInternalImpl_DestroyDirEntry,
5557   StorageInternalImpl_StreamReadAt,
5558   StorageInternalImpl_StreamWriteAt,
5559   StorageInternalImpl_StreamSetSize,
5560   StorageInternalImpl_StreamLink,
5561   StorageInternalImpl_GetTransactionSig,
5562   StorageInternalImpl_SetTransactionSig,
5563   StorageInternalImpl_LockTransaction,
5564   StorageInternalImpl_UnlockTransaction
5565 };
5566 
5567 static StorageInternalImpl* StorageInternalImpl_Construct(
5568   StorageBaseImpl* parentStorage,
5569   DWORD        openFlags,
5570   DirRef       storageDirEntry)
5571 {
5572   StorageInternalImpl* newStorage;
5573 
5574   newStorage = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(StorageInternalImpl));
5575 
5576   if (newStorage!=0)
5577   {
5578     list_init(&newStorage->base.strmHead);
5579 
5580     list_init(&newStorage->base.storageHead);
5581 
5582     /*
5583      * Initialize the virtual function table.
5584      */
5585     newStorage->base.IStorage_iface.lpVtbl = &StorageInternalImpl_Vtbl;
5586     newStorage->base.IPropertySetStorage_iface.lpVtbl = &IPropertySetStorage_Vtbl;
5587     newStorage->base.baseVtbl = &StorageInternalImpl_BaseVtbl;
5588     newStorage->base.openFlags = (openFlags & ~STGM_CREATE);
5589 
5590     newStorage->base.reverted = FALSE;
5591 
5592     newStorage->base.ref = 1;
5593 
5594     newStorage->parentStorage = parentStorage;
5595 
5596     /*
5597      * Keep a reference to the directory entry of this storage
5598      */
5599     newStorage->base.storageDirEntry = storageDirEntry;
5600 
5601     newStorage->base.create = FALSE;
5602 
5603     return newStorage;
5604   }
5605 
5606   return 0;
5607 }
5608 
5609 
5610 /************************************************************************
5611  * TransactedSnapshotImpl implementation
5612  ***********************************************************************/
5613 
5614 static DirRef TransactedSnapshotImpl_FindFreeEntry(TransactedSnapshotImpl *This)
5615 {
5616   DirRef result=This->firstFreeEntry;
5617 
5618   while (result < This->entries_size && This->entries[result].inuse)
5619     result++;
5620 
5621   if (result == This->entries_size)
5622   {
5623     ULONG new_size = This->entries_size * 2;
5624     TransactedDirEntry *new_entries;
5625 
5626     new_entries = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(TransactedDirEntry) * new_size);
5627     if (!new_entries) return DIRENTRY_NULL;
5628 
5629     memcpy(new_entries, This->entries, sizeof(TransactedDirEntry) * This->entries_size);
5630     HeapFree(GetProcessHeap(), 0, This->entries);
5631 
5632     This->entries = new_entries;
5633     This->entries_size = new_size;
5634   }
5635 
5636   This->entries[result].inuse = TRUE;
5637 
5638   This->firstFreeEntry = result+1;
5639 
5640   return result;
5641 }
5642 
5643 static DirRef TransactedSnapshotImpl_CreateStubEntry(
5644   TransactedSnapshotImpl *This, DirRef parentEntryRef)
5645 {
5646   DirRef stubEntryRef;
5647   TransactedDirEntry *entry;
5648 
5649   stubEntryRef = TransactedSnapshotImpl_FindFreeEntry(This);
5650 
5651   if (stubEntryRef != DIRENTRY_NULL)
5652   {
5653     entry = &This->entries[stubEntryRef];
5654 
5655     entry->newTransactedParentEntry = entry->transactedParentEntry = parentEntryRef;
5656 
5657     entry->read = FALSE;
5658   }
5659 
5660   return stubEntryRef;
5661 }
5662 
5663 static HRESULT TransactedSnapshotImpl_EnsureReadEntry(
5664   TransactedSnapshotImpl *This, DirRef entry)
5665 {
5666   HRESULT hr=S_OK;
5667   DirEntry data;
5668 
5669   if (!This->entries[entry].read)
5670   {
5671     hr = StorageBaseImpl_ReadDirEntry(This->transactedParent,
5672         This->entries[entry].transactedParentEntry,
5673         &data);
5674 
5675     if (SUCCEEDED(hr) && data.leftChild != DIRENTRY_NULL)
5676     {
5677       data.leftChild = TransactedSnapshotImpl_CreateStubEntry(This, data.leftChild);
5678 
5679       if (data.leftChild == DIRENTRY_NULL)
5680         hr = E_OUTOFMEMORY;
5681     }
5682 
5683     if (SUCCEEDED(hr) && data.rightChild != DIRENTRY_NULL)
5684     {
5685       data.rightChild = TransactedSnapshotImpl_CreateStubEntry(This, data.rightChild);
5686 
5687       if (data.rightChild == DIRENTRY_NULL)
5688         hr = E_OUTOFMEMORY;
5689     }
5690 
5691     if (SUCCEEDED(hr) && data.dirRootEntry != DIRENTRY_NULL)
5692     {
5693       data.dirRootEntry = TransactedSnapshotImpl_CreateStubEntry(This, data.dirRootEntry);
5694 
5695       if (data.dirRootEntry == DIRENTRY_NULL)
5696         hr = E_OUTOFMEMORY;
5697     }
5698 
5699     if (SUCCEEDED(hr))
5700     {
5701       memcpy(&This->entries[entry].data, &data, sizeof(DirEntry));
5702       This->entries[entry].read = TRUE;
5703     }
5704   }
5705 
5706   return hr;
5707 }
5708 
5709 static HRESULT TransactedSnapshotImpl_MakeStreamDirty(
5710   TransactedSnapshotImpl *This, DirRef entry)
5711 {
5712   HRESULT hr = S_OK;
5713 
5714   if (!This->entries[entry].stream_dirty)
5715   {
5716     DirEntry new_entrydata;
5717 
5718     memset(&new_entrydata, 0, sizeof(DirEntry));
5719     new_entrydata.name[0] = 'S';
5720     new_entrydata.sizeOfNameString = 1;
5721     new_entrydata.stgType = STGTY_STREAM;
5722     new_entrydata.startingBlock = BLOCK_END_OF_CHAIN;
5723     new_entrydata.leftChild = DIRENTRY_NULL;
5724     new_entrydata.rightChild = DIRENTRY_NULL;
5725     new_entrydata.dirRootEntry = DIRENTRY_NULL;
5726 
5727     hr = StorageBaseImpl_CreateDirEntry(This->scratch, &new_entrydata,
5728       &This->entries[entry].stream_entry);
5729 
5730     if (SUCCEEDED(hr) && This->entries[entry].transactedParentEntry != DIRENTRY_NULL)
5731     {
5732       hr = StorageBaseImpl_CopyStream(
5733         This->scratch, This->entries[entry].stream_entry,
5734         This->transactedParent, This->entries[entry].transactedParentEntry);
5735 
5736       if (FAILED(hr))
5737         StorageBaseImpl_DestroyDirEntry(This->scratch, This->entries[entry].stream_entry);
5738     }
5739 
5740     if (SUCCEEDED(hr))
5741       This->entries[entry].stream_dirty = TRUE;
5742 
5743     if (This->entries[entry].transactedParentEntry != DIRENTRY_NULL)
5744     {
5745       /* Since this entry is modified, and we aren't using its stream data, we
5746        * no longer care about the original entry. */
5747       DirRef delete_ref;
5748       delete_ref = TransactedSnapshotImpl_CreateStubEntry(This, This->entries[entry].transactedParentEntry);
5749 
5750       if (delete_ref != DIRENTRY_NULL)
5751         This->entries[delete_ref].deleted = TRUE;
5752 
5753       This->entries[entry].transactedParentEntry = This->entries[entry].newTransactedParentEntry = DIRENTRY_NULL;
5754     }
5755   }
5756 
5757   return hr;
5758 }
5759 
5760 /* Find the first entry in a depth-first traversal. */
5761 static DirRef TransactedSnapshotImpl_FindFirstChild(
5762   TransactedSnapshotImpl* This, DirRef parent)
5763 {
5764   DirRef cursor, prev;
5765   TransactedDirEntry *entry;
5766 
5767   cursor = parent;
5768   entry = &This->entries[cursor];
5769   while (entry->read)
5770   {
5771     if (entry->data.leftChild != DIRENTRY_NULL)
5772     {
5773       prev = cursor;
5774       cursor = entry->data.leftChild;
5775       entry = &This->entries[cursor];
5776       entry->parent = prev;
5777     }
5778     else if (entry->data.rightChild != DIRENTRY_NULL)
5779     {
5780       prev = cursor;
5781       cursor = entry->data.rightChild;
5782       entry = &This->entries[cursor];
5783       entry->parent = prev;
5784     }
5785     else if (entry->data.dirRootEntry != DIRENTRY_NULL)
5786     {
5787       prev = cursor;
5788       cursor = entry->data.dirRootEntry;
5789       entry = &This->entries[cursor];
5790       entry->parent = prev;
5791     }
5792     else
5793       break;
5794   }
5795 
5796   return cursor;
5797 }
5798 
5799 /* Find the next entry in a depth-first traversal. */
5800 static DirRef TransactedSnapshotImpl_FindNextChild(
5801   TransactedSnapshotImpl* This, DirRef current)
5802 {
5803   DirRef parent;
5804   TransactedDirEntry *parent_entry;
5805 
5806   parent = This->entries[current].parent;
5807   parent_entry = &This->entries[parent];
5808 
5809   if (parent != DIRENTRY_NULL && parent_entry->data.dirRootEntry != current)
5810   {
5811     if (parent_entry->data.rightChild != current && parent_entry->data.rightChild != DIRENTRY_NULL)
5812     {
5813       This->entries[parent_entry->data.rightChild].parent = parent;
5814       return TransactedSnapshotImpl_FindFirstChild(This, parent_entry->data.rightChild);
5815     }
5816 
5817     if (parent_entry->data.dirRootEntry != DIRENTRY_NULL)
5818     {
5819       This->entries[parent_entry->data.dirRootEntry].parent = parent;
5820       return TransactedSnapshotImpl_FindFirstChild(This, parent_entry->data.dirRootEntry);
5821     }
5822   }
5823 
5824   return parent;
5825 }
5826 
5827 /* Return TRUE if we've made a copy of this entry for committing to the parent. */
5828 static inline BOOL TransactedSnapshotImpl_MadeCopy(
5829   TransactedSnapshotImpl* This, DirRef entry)
5830 {
5831   return entry != DIRENTRY_NULL &&
5832     This->entries[entry].newTransactedParentEntry != This->entries[entry].transactedParentEntry;
5833 }
5834 
5835 /* Destroy the entries created by CopyTree. */
5836 static void TransactedSnapshotImpl_DestroyTemporaryCopy(
5837   TransactedSnapshotImpl* This, DirRef stop)
5838 {
5839   DirRef cursor;
5840   TransactedDirEntry *entry;
5841   ULARGE_INTEGER zero;
5842 
5843   zero.QuadPart = 0;
5844 
5845   if (!This->entries[This->base.storageDirEntry].read)
5846     return;
5847 
5848   cursor = This->entries[This->base.storageDirEntry].data.dirRootEntry;
5849 
5850   if (cursor == DIRENTRY_NULL)
5851     return;
5852 
5853   cursor = TransactedSnapshotImpl_FindFirstChild(This, cursor);
5854 
5855   while (cursor != DIRENTRY_NULL && cursor != stop)
5856   {
5857     if (TransactedSnapshotImpl_MadeCopy(This, cursor))
5858     {
5859       entry = &This->entries[cursor];
5860 
5861       if (entry->stream_dirty)
5862         StorageBaseImpl_StreamSetSize(This->transactedParent,
5863           entry->newTransactedParentEntry, zero);
5864 
5865       StorageBaseImpl_DestroyDirEntry(This->transactedParent,
5866         entry->newTransactedParentEntry);
5867 
5868       entry->newTransactedParentEntry = entry->transactedParentEntry;
5869     }
5870 
5871     cursor = TransactedSnapshotImpl_FindNextChild(This, cursor);
5872   }
5873 }
5874 
5875 /* Make a copy of our edited tree that we can use in the parent. */
5876 static HRESULT TransactedSnapshotImpl_CopyTree(TransactedSnapshotImpl* This)
5877 {
5878   DirRef cursor;
5879   TransactedDirEntry *entry;
5880   HRESULT hr = S_OK;
5881 
5882   cursor = This->base.storageDirEntry;
5883   entry = &This->entries[cursor];
5884   entry->parent = DIRENTRY_NULL;
5885   entry->newTransactedParentEntry = entry->transactedParentEntry;
5886 
5887   if (entry->data.dirRootEntry == DIRENTRY_NULL)
5888     return S_OK;
5889 
5890   This->entries[entry->data.dirRootEntry].parent = DIRENTRY_NULL;
5891 
5892   cursor = TransactedSnapshotImpl_FindFirstChild(This, entry->data.dirRootEntry);
5893   entry = &This->entries[cursor];
5894 
5895   while (cursor != DIRENTRY_NULL)
5896   {
5897     /* Make a copy of this entry in the transacted parent. */
5898     if (!entry->read ||
5899         (!entry->dirty && !entry->stream_dirty &&
5900          !TransactedSnapshotImpl_MadeCopy(This, entry->data.leftChild) &&
5901          !TransactedSnapshotImpl_MadeCopy(This, entry->data.rightChild) &&
5902          !TransactedSnapshotImpl_MadeCopy(This, entry->data.dirRootEntry)))
5903       entry->newTransactedParentEntry = entry->transactedParentEntry;
5904     else
5905     {
5906       DirEntry newData;
5907 
5908       memcpy(&newData, &entry->data, sizeof(DirEntry));
5909 
5910       newData.size.QuadPart = 0;
5911       newData.startingBlock = BLOCK_END_OF_CHAIN;
5912 
5913       if (newData.leftChild != DIRENTRY_NULL)
5914         newData.leftChild = This->entries[newData.leftChild].newTransactedParentEntry;
5915 
5916       if (newData.rightChild != DIRENTRY_NULL)
5917         newData.rightChild = This->entries[newData.rightChild].newTransactedParentEntry;
5918 
5919       if (newData.dirRootEntry != DIRENTRY_NULL)
5920         newData.dirRootEntry = This->entries[newData.dirRootEntry].newTransactedParentEntry;
5921 
5922       hr = StorageBaseImpl_CreateDirEntry(This->transactedParent, &newData,
5923         &entry->newTransactedParentEntry);
5924       if (FAILED(hr))
5925       {
5926         TransactedSnapshotImpl_DestroyTemporaryCopy(This, cursor);
5927         return hr;
5928       }
5929 
5930       if (entry->stream_dirty)
5931       {
5932         hr = StorageBaseImpl_CopyStream(
5933           This->transactedParent, entry->newTransactedParentEntry,
5934           This->scratch, entry->stream_entry);
5935       }
5936       else if (entry->data.size.QuadPart)
5937       {
5938         hr = StorageBaseImpl_StreamLink(
5939           This->transactedParent, entry->newTransactedParentEntry,
5940           entry->transactedParentEntry);
5941       }
5942 
5943       if (FAILED(hr))
5944       {
5945         cursor = TransactedSnapshotImpl_FindNextChild(This, cursor);
5946         TransactedSnapshotImpl_DestroyTemporaryCopy(This, cursor);
5947         return hr;
5948       }
5949     }
5950 
5951     cursor = TransactedSnapshotImpl_FindNextChild(This, cursor);
5952     entry = &This->entries[cursor];
5953   }
5954 
5955   return hr;
5956 }
5957 
5958 static HRESULT WINAPI TransactedSnapshotImpl_Commit(
5959   IStorage*            iface,
5960   DWORD                  grfCommitFlags)  /* [in] */
5961 {
5962   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*)impl_from_IStorage(iface);
5963   TransactedDirEntry *root_entry;
5964   DirRef i, dir_root_ref;
5965   DirEntry data;
5966   ULARGE_INTEGER zero;
5967   HRESULT hr;
5968   ULONG transactionSig;
5969 
5970   zero.QuadPart = 0;
5971 
5972   TRACE("(%p,%x)\n", iface, grfCommitFlags);
5973 
5974   /* Cannot commit a read-only transacted storage */
5975   if ( STGM_ACCESS_MODE( This->base.openFlags ) == STGM_READ )
5976     return STG_E_ACCESSDENIED;
5977 
5978   hr = StorageBaseImpl_LockTransaction(This->transactedParent, TRUE);
5979   if (hr == E_NOTIMPL) hr = S_OK;
5980   if (SUCCEEDED(hr))
5981   {
5982     hr = StorageBaseImpl_GetTransactionSig(This->transactedParent, &transactionSig, TRUE);
5983     if (SUCCEEDED(hr))
5984     {
5985       if (transactionSig != This->lastTransactionSig)
5986       {
5987         ERR("file was externally modified\n");
5988         hr = STG_E_NOTCURRENT;
5989       }
5990 
5991       if (SUCCEEDED(hr))
5992       {
5993         This->lastTransactionSig = transactionSig+1;
5994         hr = StorageBaseImpl_SetTransactionSig(This->transactedParent, This->lastTransactionSig);
5995       }
5996     }
5997     else if (hr == E_NOTIMPL)
5998       hr = S_OK;
5999 
6000     if (FAILED(hr)) goto end;
6001 
6002     /* To prevent data loss, we create the new structure in the file before we
6003      * delete the old one, so that in case of errors the old data is intact. We
6004      * shouldn't do this if STGC_OVERWRITE is set, but that flag should only be
6005      * needed in the rare situation where we have just enough free disk space to
6006      * overwrite the existing data. */
6007 
6008     root_entry = &This->entries[This->base.storageDirEntry];
6009 
6010     if (!root_entry->read)
6011       goto end;
6012 
6013     hr = TransactedSnapshotImpl_CopyTree(This);
6014     if (FAILED(hr)) goto end;
6015 
6016     if (root_entry->data.dirRootEntry == DIRENTRY_NULL)
6017       dir_root_ref = DIRENTRY_NULL;
6018     else
6019       dir_root_ref = This->entries[root_entry->data.dirRootEntry].newTransactedParentEntry;
6020 
6021     hr = StorageBaseImpl_Flush(This->transactedParent);
6022 
6023     /* Update the storage to use the new data in one step. */
6024     if (SUCCEEDED(hr))
6025       hr = StorageBaseImpl_ReadDirEntry(This->transactedParent,
6026         root_entry->transactedParentEntry, &data);
6027 
6028     if (SUCCEEDED(hr))
6029     {
6030       data.dirRootEntry = dir_root_ref;
6031       data.clsid = root_entry->data.clsid;
6032       data.ctime = root_entry->data.ctime;
6033       data.mtime = root_entry->data.mtime;
6034 
6035       hr = StorageBaseImpl_WriteDirEntry(This->transactedParent,
6036         root_entry->transactedParentEntry, &data);
6037     }
6038 
6039     /* Try to flush after updating the root storage, but if the flush fails, keep
6040      * going, on the theory that it'll either succeed later or the subsequent
6041      * writes will fail. */
6042     StorageBaseImpl_Flush(This->transactedParent);
6043 
6044     if (SUCCEEDED(hr))
6045     {
6046       /* Destroy the old now-orphaned data. */
6047       for (i=0; i<This->entries_size; i++)
6048       {
6049         TransactedDirEntry *entry = &This->entries[i];
6050         if (entry->inuse)
6051         {
6052           if (entry->deleted)
6053           {
6054             StorageBaseImpl_StreamSetSize(This->transactedParent,
6055               entry->transactedParentEntry, zero);
6056             StorageBaseImpl_DestroyDirEntry(This->transactedParent,
6057               entry->transactedParentEntry);
6058             memset(entry, 0, sizeof(TransactedDirEntry));
6059             This->firstFreeEntry = min(i, This->firstFreeEntry);
6060           }
6061           else if (entry->read && entry->transactedParentEntry != entry->newTransactedParentEntry)
6062           {
6063             if (entry->transactedParentEntry != DIRENTRY_NULL)
6064               StorageBaseImpl_DestroyDirEntry(This->transactedParent,
6065                 entry->transactedParentEntry);
6066             if (entry->stream_dirty)
6067             {
6068               StorageBaseImpl_StreamSetSize(This->scratch, entry->stream_entry, zero);
6069               StorageBaseImpl_DestroyDirEntry(This->scratch, entry->stream_entry);
6070               entry->stream_dirty = FALSE;
6071             }
6072             entry->dirty = FALSE;
6073             entry->transactedParentEntry = entry->newTransactedParentEntry;
6074           }
6075         }
6076       }
6077     }
6078     else
6079     {
6080       TransactedSnapshotImpl_DestroyTemporaryCopy(This, DIRENTRY_NULL);
6081     }
6082 
6083     if (SUCCEEDED(hr))
6084       hr = StorageBaseImpl_Flush(This->transactedParent);
6085 end:
6086     StorageBaseImpl_UnlockTransaction(This->transactedParent, TRUE);
6087   }
6088 
6089   TRACE("<-- %08x\n", hr);
6090   return hr;
6091 }
6092 
6093 static HRESULT WINAPI TransactedSnapshotImpl_Revert(
6094   IStorage*            iface)
6095 {
6096   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*)impl_from_IStorage(iface);
6097   ULARGE_INTEGER zero;
6098   ULONG i;
6099 
6100   TRACE("(%p)\n", iface);
6101 
6102   /* Destroy the open objects. */
6103   StorageBaseImpl_DeleteAll(&This->base);
6104 
6105   /* Clear out the scratch file. */
6106   zero.QuadPart = 0;
6107   for (i=0; i<This->entries_size; i++)
6108   {
6109     if (This->entries[i].stream_dirty)
6110     {
6111       StorageBaseImpl_StreamSetSize(This->scratch, This->entries[i].stream_entry,
6112         zero);
6113 
6114       StorageBaseImpl_DestroyDirEntry(This->scratch, This->entries[i].stream_entry);
6115     }
6116   }
6117 
6118   memset(This->entries, 0, sizeof(TransactedDirEntry) * This->entries_size);
6119 
6120   This->firstFreeEntry = 0;
6121   This->base.storageDirEntry = TransactedSnapshotImpl_CreateStubEntry(This, This->transactedParent->storageDirEntry);
6122 
6123   return S_OK;
6124 }
6125 
6126 static void TransactedSnapshotImpl_Invalidate(StorageBaseImpl* This)
6127 {
6128   if (!This->reverted)
6129   {
6130     TRACE("Storage invalidated (stg=%p)\n", This);
6131 
6132     This->reverted = TRUE;
6133 
6134     StorageBaseImpl_DeleteAll(This);
6135   }
6136 }
6137 
6138 static void TransactedSnapshotImpl_Destroy( StorageBaseImpl *iface)
6139 {
6140   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) iface;
6141 
6142   IStorage_Revert(&This->base.IStorage_iface);
6143   IStorage_Release(&This->transactedParent->IStorage_iface);
6144   IStorage_Release(&This->scratch->IStorage_iface);
6145   HeapFree(GetProcessHeap(), 0, This->entries);
6146   HeapFree(GetProcessHeap(), 0, This);
6147 }
6148 
6149 static HRESULT TransactedSnapshotImpl_Flush(StorageBaseImpl* iface)
6150 {
6151   /* We only need to flush when committing. */
6152   return S_OK;
6153 }
6154 
6155 static HRESULT TransactedSnapshotImpl_GetFilename(StorageBaseImpl* iface, LPWSTR *result)
6156 {
6157   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) iface;
6158 
6159   return StorageBaseImpl_GetFilename(This->transactedParent, result);
6160 }
6161 
6162 static HRESULT TransactedSnapshotImpl_CreateDirEntry(StorageBaseImpl *base,
6163   const DirEntry *newData, DirRef *index)
6164 {
6165   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) base;
6166   DirRef new_ref;
6167   TransactedDirEntry *new_entry;
6168 
6169   new_ref = TransactedSnapshotImpl_FindFreeEntry(This);
6170   if (new_ref == DIRENTRY_NULL)
6171     return E_OUTOFMEMORY;
6172 
6173   new_entry = &This->entries[new_ref];
6174 
6175   new_entry->newTransactedParentEntry = new_entry->transactedParentEntry = DIRENTRY_NULL;
6176   new_entry->read = TRUE;
6177   new_entry->dirty = TRUE;
6178   memcpy(&new_entry->data, newData, sizeof(DirEntry));
6179 
6180   *index = new_ref;
6181 
6182   TRACE("%s l=%x r=%x d=%x <-- %x\n", debugstr_w(newData->name), newData->leftChild, newData->rightChild, newData->dirRootEntry, *index);
6183 
6184   return S_OK;
6185 }
6186 
6187 static HRESULT TransactedSnapshotImpl_WriteDirEntry(StorageBaseImpl *base,
6188   DirRef index, const DirEntry *data)
6189 {
6190   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) base;
6191   HRESULT hr;
6192 
6193   TRACE("%x %s l=%x r=%x d=%x\n", index, debugstr_w(data->name), data->leftChild, data->rightChild, data->dirRootEntry);
6194 
6195   hr = TransactedSnapshotImpl_EnsureReadEntry(This, index);
6196   if (FAILED(hr))
6197   {
6198     TRACE("<-- %08x\n", hr);
6199     return hr;
6200   }
6201 
6202   memcpy(&This->entries[index].data, data, sizeof(DirEntry));
6203 
6204   if (index != This->base.storageDirEntry)
6205   {
6206     This->entries[index].dirty = TRUE;
6207 
6208     if (data->size.QuadPart == 0 &&
6209         This->entries[index].transactedParentEntry != DIRENTRY_NULL)
6210     {
6211       /* Since this entry is modified, and we aren't using its stream data, we
6212        * no longer care about the original entry. */
6213       DirRef delete_ref;
6214       delete_ref = TransactedSnapshotImpl_CreateStubEntry(This, This->entries[index].transactedParentEntry);
6215 
6216       if (delete_ref != DIRENTRY_NULL)
6217         This->entries[delete_ref].deleted = TRUE;
6218 
6219       This->entries[index].transactedParentEntry = This->entries[index].newTransactedParentEntry = DIRENTRY_NULL;
6220     }
6221   }
6222   TRACE("<-- S_OK\n");
6223   return S_OK;
6224 }
6225 
6226 static HRESULT TransactedSnapshotImpl_ReadDirEntry(StorageBaseImpl *base,
6227   DirRef index, DirEntry *data)
6228 {
6229   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) base;
6230   HRESULT hr;
6231 
6232   hr = TransactedSnapshotImpl_EnsureReadEntry(This, index);
6233   if (FAILED(hr))
6234   {
6235     TRACE("<-- %08x\n", hr);
6236     return hr;
6237   }
6238 
6239   memcpy(data, &This->entries[index].data, sizeof(DirEntry));
6240 
6241   TRACE("%x %s l=%x r=%x d=%x\n", index, debugstr_w(data->name), data->leftChild, data->rightChild, data->dirRootEntry);
6242 
6243   return S_OK;
6244 }
6245 
6246 static HRESULT TransactedSnapshotImpl_DestroyDirEntry(StorageBaseImpl *base,
6247   DirRef index)
6248 {
6249   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) base;
6250 
6251   if (This->entries[index].transactedParentEntry == DIRENTRY_NULL ||
6252       This->entries[index].data.size.QuadPart != 0)
6253   {
6254     /* If we deleted this entry while it has stream data. We must have left the
6255      * data because some other entry is using it, and we need to leave the
6256      * original entry alone. */
6257     memset(&This->entries[index], 0, sizeof(TransactedDirEntry));
6258     This->firstFreeEntry = min(index, This->firstFreeEntry);
6259   }
6260   else
6261   {
6262     This->entries[index].deleted = TRUE;
6263   }
6264 
6265   return S_OK;
6266 }
6267 
6268 static HRESULT TransactedSnapshotImpl_StreamReadAt(StorageBaseImpl *base,
6269   DirRef index, ULARGE_INTEGER offset, ULONG size, void *buffer, ULONG *bytesRead)
6270 {
6271   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) base;
6272 
6273   if (This->entries[index].stream_dirty)
6274   {
6275     return StorageBaseImpl_StreamReadAt(This->scratch,
6276         This->entries[index].stream_entry, offset, size, buffer, bytesRead);
6277   }
6278   else if (This->entries[index].transactedParentEntry == DIRENTRY_NULL)
6279   {
6280     /* This stream doesn't live in the parent, and we haven't allocated storage
6281      * for it yet */
6282     *bytesRead = 0;
6283     return S_OK;
6284   }
6285   else
6286   {
6287     return StorageBaseImpl_StreamReadAt(This->transactedParent,
6288         This->entries[index].transactedParentEntry, offset, size, buffer, bytesRead);
6289   }
6290 }
6291 
6292 static HRESULT TransactedSnapshotImpl_StreamWriteAt(StorageBaseImpl *base,
6293   DirRef index, ULARGE_INTEGER offset, ULONG size, const void *buffer, ULONG *bytesWritten)
6294 {
6295   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) base;
6296   HRESULT hr;
6297 
6298   hr = TransactedSnapshotImpl_EnsureReadEntry(This, index);
6299   if (FAILED(hr))
6300   {
6301     TRACE("<-- %08x\n", hr);
6302     return hr;
6303   }
6304 
6305   hr = TransactedSnapshotImpl_MakeStreamDirty(This, index);
6306   if (FAILED(hr))
6307   {
6308     TRACE("<-- %08x\n", hr);
6309     return hr;
6310   }
6311 
6312   hr = StorageBaseImpl_StreamWriteAt(This->scratch,
6313     This->entries[index].stream_entry, offset, size, buffer, bytesWritten);
6314 
6315   if (SUCCEEDED(hr) && size != 0)
6316     This->entries[index].data.size.QuadPart = max(
6317         This->entries[index].data.size.QuadPart,
6318         offset.QuadPart + size);
6319 
6320   TRACE("<-- %08x\n", hr);
6321   return hr;
6322 }
6323 
6324 static HRESULT TransactedSnapshotImpl_StreamSetSize(StorageBaseImpl *base,
6325   DirRef index, ULARGE_INTEGER newsize)
6326 {
6327   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) base;
6328   HRESULT hr;
6329 
6330   hr = TransactedSnapshotImpl_EnsureReadEntry(This, index);
6331   if (FAILED(hr))
6332   {
6333     TRACE("<-- %08x\n", hr);
6334     return hr;
6335   }
6336 
6337   if (This->entries[index].data.size.QuadPart == newsize.QuadPart)
6338     return S_OK;
6339 
6340   if (newsize.QuadPart == 0)
6341   {
6342     /* Destroy any parent references or entries in the scratch file. */
6343     if (This->entries[index].stream_dirty)
6344     {
6345       ULARGE_INTEGER zero;
6346       zero.QuadPart = 0;
6347       StorageBaseImpl_StreamSetSize(This->scratch,
6348         This->entries[index].stream_entry, zero);
6349       StorageBaseImpl_DestroyDirEntry(This->scratch,
6350         This->entries[index].stream_entry);
6351       This->entries[index].stream_dirty = FALSE;
6352     }
6353     else if (This->entries[index].transactedParentEntry != DIRENTRY_NULL)
6354     {
6355       DirRef delete_ref;
6356       delete_ref = TransactedSnapshotImpl_CreateStubEntry(This, This->entries[index].transactedParentEntry);
6357 
6358       if (delete_ref != DIRENTRY_NULL)
6359         This->entries[delete_ref].deleted = TRUE;
6360 
6361       This->entries[index].transactedParentEntry = This->entries[index].newTransactedParentEntry = DIRENTRY_NULL;
6362     }
6363   }
6364   else
6365   {
6366     hr = TransactedSnapshotImpl_MakeStreamDirty(This, index);
6367     if (FAILED(hr)) return hr;
6368 
6369     hr = StorageBaseImpl_StreamSetSize(This->scratch,
6370       This->entries[index].stream_entry, newsize);
6371   }
6372 
6373   if (SUCCEEDED(hr))
6374     This->entries[index].data.size = newsize;
6375 
6376   TRACE("<-- %08x\n", hr);
6377   return hr;
6378 }
6379 
6380 static HRESULT TransactedSnapshotImpl_StreamLink(StorageBaseImpl *base,
6381   DirRef dst, DirRef src)
6382 {
6383   TransactedSnapshotImpl* This = (TransactedSnapshotImpl*) base;
6384   HRESULT hr;
6385   TransactedDirEntry *dst_entry, *src_entry;
6386 
6387   hr = TransactedSnapshotImpl_EnsureReadEntry(This, src);
6388   if (FAILED(hr))
6389   {
6390     TRACE("<-- %08x\n", hr);
6391     return hr;
6392   }
6393 
6394   hr = TransactedSnapshotImpl_EnsureReadEntry(This, dst);
6395   if (FAILED(hr))
6396   {
6397     TRACE("<-- %08x\n", hr);
6398     return hr;
6399   }
6400 
6401   dst_entry = &This->entries[dst];
6402   src_entry = &This->entries[src];
6403 
6404   dst_entry->stream_dirty = src_entry->stream_dirty;
6405   dst_entry->stream_entry = src_entry->stream_entry;
6406   dst_entry->transactedParentEntry = src_entry->transactedParentEntry;
6407   dst_entry->newTransactedParentEntry = src_entry->newTransactedParentEntry;
6408   dst_entry->data.size = src_entry->data.size;
6409 
6410   return S_OK;
6411 }
6412 
6413 static HRESULT TransactedSnapshotImpl_GetTransactionSig(StorageBaseImpl *base,
6414   ULONG* result, BOOL refresh)
6415 {
6416   return E_NOTIMPL;
6417 }
6418 
6419 static HRESULT TransactedSnapshotImpl_SetTransactionSig(StorageBaseImpl *base,
6420   ULONG value)
6421 {
6422   return E_NOTIMPL;
6423 }
6424 
6425 static HRESULT TransactedSnapshotImpl_LockTransaction(StorageBaseImpl *base, BOOL write)
6426 {
6427   return E_NOTIMPL;
6428 }
6429 
6430 static HRESULT TransactedSnapshotImpl_UnlockTransaction(StorageBaseImpl *base, BOOL write)
6431 {
6432   return E_NOTIMPL;
6433 }
6434 
6435 static const IStorageVtbl TransactedSnapshotImpl_Vtbl =
6436 {
6437     StorageBaseImpl_QueryInterface,
6438     StorageBaseImpl_AddRef,
6439     StorageBaseImpl_Release,
6440     StorageBaseImpl_CreateStream,
6441     StorageBaseImpl_OpenStream,
6442     StorageBaseImpl_CreateStorage,
6443     StorageBaseImpl_OpenStorage,
6444     StorageBaseImpl_CopyTo,
6445     StorageBaseImpl_MoveElementTo,
6446     TransactedSnapshotImpl_Commit,
6447     TransactedSnapshotImpl_Revert,
6448     StorageBaseImpl_EnumElements,
6449     StorageBaseImpl_DestroyElement,
6450     StorageBaseImpl_RenameElement,
6451     StorageBaseImpl_SetElementTimes,
6452     StorageBaseImpl_SetClass,
6453     StorageBaseImpl_SetStateBits,
6454     StorageBaseImpl_Stat
6455 };
6456 
6457 static const StorageBaseImplVtbl TransactedSnapshotImpl_BaseVtbl =
6458 {
6459   TransactedSnapshotImpl_Destroy,
6460   TransactedSnapshotImpl_Invalidate,
6461   TransactedSnapshotImpl_Flush,
6462   TransactedSnapshotImpl_GetFilename,
6463   TransactedSnapshotImpl_CreateDirEntry,
6464   TransactedSnapshotImpl_WriteDirEntry,
6465   TransactedSnapshotImpl_ReadDirEntry,
6466   TransactedSnapshotImpl_DestroyDirEntry,
6467   TransactedSnapshotImpl_StreamReadAt,
6468   TransactedSnapshotImpl_StreamWriteAt,
6469   TransactedSnapshotImpl_StreamSetSize,
6470   TransactedSnapshotImpl_StreamLink,
6471   TransactedSnapshotImpl_GetTransactionSig,
6472   TransactedSnapshotImpl_SetTransactionSig,
6473   TransactedSnapshotImpl_LockTransaction,
6474   TransactedSnapshotImpl_UnlockTransaction
6475 };
6476 
6477 static HRESULT TransactedSnapshotImpl_Construct(StorageBaseImpl *parentStorage,
6478   TransactedSnapshotImpl** result)
6479 {
6480   HRESULT hr;
6481 
6482   *result = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(TransactedSnapshotImpl));
6483   if (*result)
6484   {
6485     IStorage *scratch;
6486 
6487     (*result)->base.IStorage_iface.lpVtbl = &TransactedSnapshotImpl_Vtbl;
6488 
6489     /* This is OK because the property set storage functions use the IStorage functions. */
6490     (*result)->base.IPropertySetStorage_iface.lpVtbl = parentStorage->IPropertySetStorage_iface.lpVtbl;
6491     (*result)->base.baseVtbl = &TransactedSnapshotImpl_BaseVtbl;
6492 
6493     list_init(&(*result)->base.strmHead);
6494 
6495     list_init(&(*result)->base.storageHead);
6496 
6497     (*result)->base.ref = 1;
6498 
6499     (*result)->base.openFlags = parentStorage->openFlags;
6500 
6501     /* This cannot fail, except with E_NOTIMPL in which case we don't care */
6502     StorageBaseImpl_GetTransactionSig(parentStorage, &(*result)->lastTransactionSig, FALSE);
6503 
6504     /* Create a new temporary storage to act as the scratch file. */
6505     hr = StgCreateDocfile(NULL, STGM_READWRITE|STGM_SHARE_EXCLUSIVE|STGM_CREATE|STGM_DELETEONRELEASE,
6506         0, &scratch);
6507     (*result)->scratch = impl_from_IStorage(scratch);
6508 
6509     if (SUCCEEDED(hr))
6510     {
6511         ULONG num_entries = 20;
6512 
6513         (*result)->entries = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(TransactedDirEntry) * num_entries);
6514         (*result)->entries_size = num_entries;
6515         (*result)->firstFreeEntry = 0;
6516 
6517         if ((*result)->entries)
6518         {
6519             /* parentStorage already has 1 reference, which we take over here. */
6520             (*result)->transactedParent = parentStorage;
6521 
6522             parentStorage->transactedChild = &(*result)->base;
6523 
6524             (*result)->base.storageDirEntry = TransactedSnapshotImpl_CreateStubEntry(*result, parentStorage->storageDirEntry);
6525         }
6526         else
6527         {
6528             IStorage_Release(scratch);
6529 
6530             hr = E_OUTOFMEMORY;
6531         }
6532     }
6533 
6534     if (FAILED(hr)) HeapFree(GetProcessHeap(), 0, *result);
6535 
6536     return hr;
6537   }
6538   else
6539     return E_OUTOFMEMORY;
6540 }
6541 
6542 
6543 /************************************************************************
6544  * TransactedSharedImpl implementation
6545  ***********************************************************************/
6546 
6547 static void TransactedSharedImpl_Invalidate(StorageBaseImpl* This)
6548 {
6549   if (!This->reverted)
6550   {
6551     TRACE("Storage invalidated (stg=%p)\n", This);
6552 
6553     This->reverted = TRUE;
6554 
6555     StorageBaseImpl_DeleteAll(This);
6556   }
6557 }
6558 
6559 static void TransactedSharedImpl_Destroy( StorageBaseImpl *iface)
6560 {
6561   TransactedSharedImpl* This = (TransactedSharedImpl*) iface;
6562 
6563   TransactedSharedImpl_Invalidate(&This->base);
6564   IStorage_Release(&This->transactedParent->IStorage_iface);
6565   IStorage_Release(&This->scratch->base.IStorage_iface);
6566   HeapFree(GetProcessHeap(), 0, This);
6567 }
6568 
6569 static HRESULT TransactedSharedImpl_Flush(StorageBaseImpl* iface)
6570 {
6571   /* We only need to flush when committing. */
6572   return S_OK;
6573 }
6574 
6575 static HRESULT TransactedSharedImpl_GetFilename(StorageBaseImpl* iface, LPWSTR *result)
6576 {
6577   TransactedSharedImpl* This = (TransactedSharedImpl*) iface;
6578 
6579   return StorageBaseImpl_GetFilename(This->transactedParent, result);
6580 }
6581 
6582 static HRESULT TransactedSharedImpl_CreateDirEntry(StorageBaseImpl *base,
6583   const DirEntry *newData, DirRef *index)
6584 {
6585   TransactedSharedImpl* This = (TransactedSharedImpl*) base;
6586 
6587   return StorageBaseImpl_CreateDirEntry(&This->scratch->base,
6588     newData, index);
6589 }
6590 
6591 static HRESULT TransactedSharedImpl_WriteDirEntry(StorageBaseImpl *base,
6592   DirRef index, const DirEntry *data)
6593 {
6594   TransactedSharedImpl* This = (TransactedSharedImpl*) base;
6595 
6596   return StorageBaseImpl_WriteDirEntry(&This->scratch->base,
6597     index, data);
6598 }
6599 
6600 static HRESULT TransactedSharedImpl_ReadDirEntry(StorageBaseImpl *base,
6601   DirRef index, DirEntry *data)
6602 {
6603   TransactedSharedImpl* This = (TransactedSharedImpl*) base;
6604 
6605   return StorageBaseImpl_ReadDirEntry(&This->scratch->base,
6606     index, data);
6607 }
6608 
6609 static HRESULT TransactedSharedImpl_DestroyDirEntry(StorageBaseImpl *base,
6610   DirRef index)
6611 {
6612   TransactedSharedImpl* This = (TransactedSharedImpl*) base;
6613 
6614   return StorageBaseImpl_DestroyDirEntry(&This->scratch->base,
6615     index);
6616 }
6617 
6618 static HRESULT TransactedSharedImpl_StreamReadAt(StorageBaseImpl *base,
6619   DirRef index, ULARGE_INTEGER offset, ULONG size, void *buffer, ULONG *bytesRead)
6620 {
6621   TransactedSharedImpl* This = (TransactedSharedImpl*) base;
6622 
6623   return StorageBaseImpl_StreamReadAt(&This->scratch->base,
6624     index, offset, size, buffer, bytesRead);
6625 }
6626 
6627 static HRESULT TransactedSharedImpl_StreamWriteAt(StorageBaseImpl *base,
6628   DirRef index, ULARGE_INTEGER offset, ULONG size, const void *buffer, ULONG *bytesWritten)
6629 {
6630   TransactedSharedImpl* This = (TransactedSharedImpl*) base;
6631 
6632   return StorageBaseImpl_StreamWriteAt(&This->scratch->base,
6633     index, offset, size, buffer, bytesWritten);
6634 }
6635 
6636 static HRESULT TransactedSharedImpl_StreamSetSize(StorageBaseImpl *base,
6637   DirRef index, ULARGE_INTEGER newsize)
6638 {
6639   TransactedSharedImpl* This = (TransactedSharedImpl*) base;
6640 
6641   return StorageBaseImpl_StreamSetSize(&This->scratch->base,
6642     index, newsize);
6643 }
6644 
6645 static HRESULT TransactedSharedImpl_StreamLink(StorageBaseImpl *base,
6646   DirRef dst, DirRef src)
6647 {
6648   TransactedSharedImpl* This = (TransactedSharedImpl*) base;
6649 
6650   return StorageBaseImpl_StreamLink(&This->scratch->base,
6651     dst, src);
6652 }
6653 
6654 static HRESULT TransactedSharedImpl_GetTransactionSig(StorageBaseImpl *base,
6655   ULONG* result, BOOL refresh)
6656 {
6657   return E_NOTIMPL;
6658 }
6659 
6660 static HRESULT TransactedSharedImpl_SetTransactionSig(StorageBaseImpl *base,
6661   ULONG value)
6662 {
6663   return E_NOTIMPL;
6664 }
6665 
6666 static HRESULT TransactedSharedImpl_LockTransaction(StorageBaseImpl *base, BOOL write)
6667 {
6668   return E_NOTIMPL;
6669 }
6670 
6671 static HRESULT TransactedSharedImpl_UnlockTransaction(StorageBaseImpl *base, BOOL write)
6672 {
6673   return E_NOTIMPL;
6674 }
6675 
6676 static HRESULT WINAPI TransactedSharedImpl_Commit(
6677   IStorage*            iface,
6678   DWORD                  grfCommitFlags)  /* [in] */
6679 {
6680   TransactedSharedImpl* This = (TransactedSharedImpl*)impl_from_IStorage(iface);
6681   DirRef new_storage_ref, prev_storage_ref;
6682   DirEntry src_data, dst_data;
6683   HRESULT hr;
6684   ULONG transactionSig;
6685 
6686   TRACE("(%p,%x)\n", iface, grfCommitFlags);
6687 
6688   /* Cannot commit a read-only transacted storage */
6689   if ( STGM_ACCESS_MODE( This->base.openFlags ) == STGM_READ )
6690     return STG_E_ACCESSDENIED;
6691 
6692   hr = StorageBaseImpl_LockTransaction(This->transactedParent, TRUE);
6693   if (hr == E_NOTIMPL) hr = S_OK;
6694   if (SUCCEEDED(hr))
6695   {
6696     hr = StorageBaseImpl_GetTransactionSig(This->transactedParent, &transactionSig, TRUE);
6697     if (SUCCEEDED(hr))
6698     {
6699       if ((grfCommitFlags & STGC_ONLYIFCURRENT) && transactionSig != This->lastTransactionSig)
6700         hr = STG_E_NOTCURRENT;
6701 
6702       if (SUCCEEDED(hr))
6703         hr = StorageBaseImpl_SetTransactionSig(This->transactedParent, transactionSig+1);
6704     }
6705     else if (hr == E_NOTIMPL)
6706       hr = S_OK;
6707 
6708     if (SUCCEEDED(hr))
6709       hr = StorageBaseImpl_ReadDirEntry(&This->scratch->base, This->scratch->base.storageDirEntry, &src_data);
6710 
6711     /* FIXME: If we're current, we should be able to copy only the changes in scratch. */
6712     if (SUCCEEDED(hr))
6713       hr = StorageBaseImpl_DupStorageTree(This->transactedParent, &new_storage_ref, &This->scratch->base, src_data.dirRootEntry);
6714 
6715     if (SUCCEEDED(hr))
6716       hr = StorageBaseImpl_Flush(This->transactedParent);
6717 
6718     if (SUCCEEDED(hr))
6719       hr = StorageBaseImpl_ReadDirEntry(This->transactedParent, This->transactedParent->storageDirEntry, &dst_data);
6720 
6721     if (SUCCEEDED(hr))
6722     {
6723       prev_storage_ref = dst_data.dirRootEntry;
6724       dst_data.dirRootEntry = new_storage_ref;
6725       dst_data.clsid = src_data.clsid;
6726       dst_data.ctime = src_data.ctime;
6727       dst_data.mtime = src_data.mtime;
6728       hr = StorageBaseImpl_WriteDirEntry(This->transactedParent, This->transactedParent->storageDirEntry, &dst_data);
6729     }
6730 
6731     if (SUCCEEDED(hr))
6732     {
6733       /* Try to flush after updating the root storage, but if the flush fails, keep
6734        * going, on the theory that it'll either succeed later or the subsequent
6735        * writes will fail. */
6736       StorageBaseImpl_Flush(This->transactedParent);
6737 
6738       hr = StorageBaseImpl_DeleteStorageTree(This->transactedParent, prev_storage_ref, TRUE);
6739     }
6740 
6741     if (SUCCEEDED(hr))
6742       hr = StorageBaseImpl_Flush(This->transactedParent);
6743 
6744     StorageBaseImpl_UnlockTransaction(This->transactedParent, TRUE);
6745 
6746     if (SUCCEEDED(hr))
6747       hr = IStorage_Commit(&This->scratch->base.IStorage_iface, STGC_DEFAULT);
6748 
6749     if (SUCCEEDED(hr))
6750     {
6751       This->lastTransactionSig = transactionSig+1;
6752     }
6753   }
6754   TRACE("<-- %08x\n", hr);
6755   return hr;
6756 }
6757 
6758 static HRESULT WINAPI TransactedSharedImpl_Revert(
6759   IStorage*            iface)
6760 {
6761   TransactedSharedImpl* This = (TransactedSharedImpl*)impl_from_IStorage(iface);
6762 
6763   TRACE("(%p)\n", iface);
6764 
6765   /* Destroy the open objects. */
6766   StorageBaseImpl_DeleteAll(&This->base);
6767 
6768   return IStorage_Revert(&This->scratch->base.IStorage_iface);
6769 }
6770 
6771 static const IStorageVtbl TransactedSharedImpl_Vtbl =
6772 {
6773     StorageBaseImpl_QueryInterface,
6774     StorageBaseImpl_AddRef,
6775     StorageBaseImpl_Release,
6776     StorageBaseImpl_CreateStream,
6777     StorageBaseImpl_OpenStream,
6778     StorageBaseImpl_CreateStorage,
6779     StorageBaseImpl_OpenStorage,
6780     StorageBaseImpl_CopyTo,
6781     StorageBaseImpl_MoveElementTo,
6782     TransactedSharedImpl_Commit,
6783     TransactedSharedImpl_Revert,
6784     StorageBaseImpl_EnumElements,
6785     StorageBaseImpl_DestroyElement,
6786     StorageBaseImpl_RenameElement,
6787     StorageBaseImpl_SetElementTimes,
6788     StorageBaseImpl_SetClass,
6789     StorageBaseImpl_SetStateBits,
6790     StorageBaseImpl_Stat
6791 };
6792 
6793 static const StorageBaseImplVtbl TransactedSharedImpl_BaseVtbl =
6794 {
6795   TransactedSharedImpl_Destroy,
6796   TransactedSharedImpl_Invalidate,
6797   TransactedSharedImpl_Flush,
6798   TransactedSharedImpl_GetFilename,
6799   TransactedSharedImpl_CreateDirEntry,
6800   TransactedSharedImpl_WriteDirEntry,
6801   TransactedSharedImpl_ReadDirEntry,
6802   TransactedSharedImpl_DestroyDirEntry,
6803   TransactedSharedImpl_StreamReadAt,
6804   TransactedSharedImpl_StreamWriteAt,
6805   TransactedSharedImpl_StreamSetSize,
6806   TransactedSharedImpl_StreamLink,
6807   TransactedSharedImpl_GetTransactionSig,
6808   TransactedSharedImpl_SetTransactionSig,
6809   TransactedSharedImpl_LockTransaction,
6810   TransactedSharedImpl_UnlockTransaction
6811 };
6812 
6813 static HRESULT TransactedSharedImpl_Construct(StorageBaseImpl *parentStorage,
6814   TransactedSharedImpl** result)
6815 {
6816   HRESULT hr;
6817 
6818   *result = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(TransactedSharedImpl));
6819   if (*result)
6820   {
6821     IStorage *scratch;
6822 
6823     (*result)->base.IStorage_iface.lpVtbl = &TransactedSharedImpl_Vtbl;
6824 
6825     /* This is OK because the property set storage functions use the IStorage functions. */
6826     (*result)->base.IPropertySetStorage_iface.lpVtbl = parentStorage->IPropertySetStorage_iface.lpVtbl;
6827     (*result)->base.baseVtbl = &TransactedSharedImpl_BaseVtbl;
6828 
6829     list_init(&(*result)->base.strmHead);
6830 
6831     list_init(&(*result)->base.storageHead);
6832 
6833     (*result)->base.ref = 1;
6834 
6835     (*result)->base.openFlags = parentStorage->openFlags;
6836 
6837     hr = StorageBaseImpl_LockTransaction(parentStorage, FALSE);
6838 
6839     if (SUCCEEDED(hr))
6840     {
6841       STGOPTIONS stgo;
6842 
6843       /* This cannot fail, except with E_NOTIMPL in which case we don't care */
6844       StorageBaseImpl_GetTransactionSig(parentStorage, &(*result)->lastTransactionSig, FALSE);
6845 
6846       stgo.usVersion = 1;
6847       stgo.reserved = 0;
6848       stgo.ulSectorSize = 4096;
6849       stgo.pwcsTemplateFile = NULL;
6850 
6851       /* Create a new temporary storage to act as the scratch file. */
6852       hr = StgCreateStorageEx(NULL, STGM_READWRITE|STGM_SHARE_EXCLUSIVE|STGM_CREATE|STGM_DELETEONRELEASE|STGM_TRANSACTED,
6853           STGFMT_DOCFILE, 0, &stgo, NULL, &IID_IStorage, (void**)&scratch);
6854       (*result)->scratch = (TransactedSnapshotImpl*)impl_from_IStorage(scratch);
6855 
6856       if (SUCCEEDED(hr))
6857       {
6858         hr = StorageBaseImpl_CopyStorageTree(&(*result)->scratch->base, (*result)->scratch->base.storageDirEntry,
6859           parentStorage, parentStorage->storageDirEntry);
6860 
6861         if (SUCCEEDED(hr))
6862         {
6863           hr = IStorage_Commit(scratch, STGC_DEFAULT);
6864 
6865           (*result)->base.storageDirEntry = (*result)->scratch->base.storageDirEntry;
6866           (*result)->transactedParent = parentStorage;
6867         }
6868 
6869         if (FAILED(hr))
6870           IStorage_Release(scratch);
6871       }
6872 
6873       StorageBaseImpl_UnlockTransaction(parentStorage, FALSE);
6874     }
6875 
6876     if (FAILED(hr)) HeapFree(GetProcessHeap(), 0, *result);
6877 
6878     return hr;
6879   }
6880   else
6881     return E_OUTOFMEMORY;
6882 }
6883 
6884 static HRESULT Storage_ConstructTransacted(StorageBaseImpl *parentStorage,
6885   BOOL toplevel, StorageBaseImpl** result)
6886 {
6887   static int fixme_flags=STGM_NOSCRATCH|STGM_NOSNAPSHOT;
6888 
6889   if (parentStorage->openFlags & fixme_flags)
6890   {
6891     fixme_flags &= ~parentStorage->openFlags;
6892     FIXME("Unimplemented flags %x\n", parentStorage->openFlags);
6893   }
6894 
6895   if (toplevel && !(parentStorage->openFlags & STGM_NOSNAPSHOT) &&
6896       STGM_SHARE_MODE(parentStorage->openFlags) != STGM_SHARE_DENY_WRITE &&
6897       STGM_SHARE_MODE(parentStorage->openFlags) != STGM_SHARE_EXCLUSIVE)
6898   {
6899     /* Need to create a temp file for the snapshot */
6900     return TransactedSharedImpl_Construct(parentStorage, (TransactedSharedImpl**)result);
6901   }
6902 
6903   return TransactedSnapshotImpl_Construct(parentStorage,
6904     (TransactedSnapshotImpl**)result);
6905 }
6906 
6907 static HRESULT Storage_Construct(
6908   HANDLE       hFile,
6909   LPCOLESTR    pwcsName,
6910   ILockBytes*  pLkbyt,
6911   DWORD        openFlags,
6912   BOOL         fileBased,
6913   BOOL         create,
6914   ULONG        sector_size,
6915   StorageBaseImpl** result)
6916 {
6917   StorageImpl *newStorage;
6918   StorageBaseImpl *newTransactedStorage;
6919   HRESULT hr;
6920 
6921   hr = StorageImpl_Construct(hFile, pwcsName, pLkbyt, openFlags, fileBased, create, sector_size, &newStorage);
6922   if (FAILED(hr)) goto end;
6923 
6924   if (openFlags & STGM_TRANSACTED)
6925   {
6926     hr = Storage_ConstructTransacted(&newStorage->base, TRUE, &newTransactedStorage);
6927     if (FAILED(hr))
6928       IStorage_Release(&newStorage->base.IStorage_iface);
6929     else
6930       *result = newTransactedStorage;
6931   }
6932   else
6933     *result = &newStorage->base;
6934 
6935 end:
6936   return hr;
6937 }
6938 
6939 
6940 /************************************************************************
6941  * StorageUtl helper functions
6942  ***********************************************************************/
6943 
6944 void StorageUtl_ReadWord(const BYTE* buffer, ULONG offset, WORD* value)
6945 {
6946   WORD tmp;
6947 
6948   memcpy(&tmp, buffer+offset, sizeof(WORD));
6949   *value = lendian16toh(tmp);
6950 }
6951 
6952 void StorageUtl_WriteWord(BYTE* buffer, ULONG offset, WORD value)
6953 {
6954   value = htole16(value);
6955   memcpy(buffer+offset, &value, sizeof(WORD));
6956 }
6957 
6958 void StorageUtl_ReadDWord(const BYTE* buffer, ULONG offset, DWORD* value)
6959 {
6960   DWORD tmp;
6961 
6962   memcpy(&tmp, buffer+offset, sizeof(DWORD));
6963   *value = lendian32toh(tmp);
6964 }
6965 
6966 void StorageUtl_WriteDWord(BYTE* buffer, ULONG offset, DWORD value)
6967 {
6968   value = htole32(value);
6969   memcpy(buffer+offset, &value, sizeof(DWORD));
6970 }
6971 
6972 void StorageUtl_ReadULargeInteger(const BYTE* buffer, ULONG offset,
6973  ULARGE_INTEGER* value)
6974 {
6975 #ifdef WORDS_BIGENDIAN
6976     ULARGE_INTEGER tmp;
6977 
6978     memcpy(&tmp, buffer + offset, sizeof(ULARGE_INTEGER));
6979     value->u.LowPart = htole32(tmp.u.HighPart);
6980     value->u.HighPart = htole32(tmp.u.LowPart);
6981 #else
6982     memcpy(value, buffer + offset, sizeof(ULARGE_INTEGER));
6983 #endif
6984 }
6985 
6986 void StorageUtl_WriteULargeInteger(BYTE* buffer, ULONG offset,
6987  const ULARGE_INTEGER *value)
6988 {
6989 #ifdef WORDS_BIGENDIAN
6990     ULARGE_INTEGER tmp;
6991 
6992     tmp.u.LowPart = htole32(value->u.HighPart);
6993     tmp.u.HighPart = htole32(value->u.LowPart);
6994     memcpy(buffer + offset, &tmp, sizeof(ULARGE_INTEGER));
6995 #else
6996     memcpy(buffer + offset, value, sizeof(ULARGE_INTEGER));
6997 #endif
6998 }
6999 
7000 void StorageUtl_ReadGUID(const BYTE* buffer, ULONG offset, GUID* value)
7001 {
7002   StorageUtl_ReadDWord(buffer, offset,   &(value->Data1));
7003   StorageUtl_ReadWord(buffer,  offset+4, &(value->Data2));
7004   StorageUtl_ReadWord(buffer,  offset+6, &(value->Data3));
7005 
7006   memcpy(value->Data4, buffer+offset+8, sizeof(value->Data4));
7007 }
7008 
7009 void StorageUtl_WriteGUID(BYTE* buffer, ULONG offset, const GUID* value)
7010 {
7011   StorageUtl_WriteDWord(buffer, offset,   value->Data1);
7012   StorageUtl_WriteWord(buffer,  offset+4, value->Data2);
7013   StorageUtl_WriteWord(buffer,  offset+6, value->Data3);
7014 
7015   memcpy(buffer+offset+8, value->Data4, sizeof(value->Data4));
7016 }
7017 
7018 void StorageUtl_CopyDirEntryToSTATSTG(
7019   StorageBaseImpl*      storage,
7020   STATSTG*              destination,
7021   const DirEntry*       source,
7022   int                   statFlags)
7023 {
7024   /*
7025    * The copy of the string occurs only when the flag is not set
7026    */
7027   if (!(statFlags & STATFLAG_NONAME) && source->stgType == STGTY_ROOT)
7028   {
7029     /* Use the filename for the root storage. */
7030     destination->pwcsName = 0;
7031     StorageBaseImpl_GetFilename(storage, &destination->pwcsName);
7032   }
7033   else if( ((statFlags & STATFLAG_NONAME) != 0) ||
7034        (source->name[0] == 0) )
7035   {
7036     destination->pwcsName = 0;
7037   }
7038   else
7039   {
7040     destination->pwcsName =
7041       CoTaskMemAlloc((lstrlenW(source->name)+1)*sizeof(WCHAR));
7042 
7043     strcpyW(destination->pwcsName, source->name);
7044   }
7045 
7046   switch (source->stgType)
7047   {
7048     case STGTY_STORAGE:
7049     case STGTY_ROOT:
7050       destination->type = STGTY_STORAGE;
7051       break;
7052     case STGTY_STREAM:
7053       destination->type = STGTY_STREAM;
7054       break;
7055     default:
7056       destination->type = STGTY_STREAM;
7057       break;
7058   }
7059 
7060   destination->cbSize            = source->size;
7061 /*
7062   currentReturnStruct->mtime     = {0}; TODO
7063   currentReturnStruct->ctime     = {0};
7064   currentReturnStruct->atime     = {0};
7065 */
7066   destination->grfMode           = 0;
7067   destination->grfLocksSupported = 0;
7068   destination->clsid             = source->clsid;
7069   destination->grfStateBits      = 0;
7070   destination->reserved          = 0;
7071 }
7072 
7073 
7074 /************************************************************************
7075  * BlockChainStream implementation
7076  ***********************************************************************/
7077 
7078 /******************************************************************************
7079  *      BlockChainStream_GetHeadOfChain
7080  *
7081  * Returns the head of this stream chain.
7082  * Some special chains don't have directory entries, their heads are kept in
7083  * This->headOfStreamPlaceHolder.
7084  *
7085  */
7086 static ULONG BlockChainStream_GetHeadOfChain(BlockChainStream* This)
7087 {
7088   DirEntry  chainEntry;
7089   HRESULT   hr;
7090 
7091   if (This->headOfStreamPlaceHolder != 0)
7092     return *(This->headOfStreamPlaceHolder);
7093 
7094   if (This->ownerDirEntry != DIRENTRY_NULL)
7095   {
7096     hr = StorageImpl_ReadDirEntry(
7097                       This->parentStorage,
7098                       This->ownerDirEntry,
7099                       &chainEntry);
7100 
7101     if (SUCCEEDED(hr) && chainEntry.startingBlock < BLOCK_FIRST_SPECIAL)
7102       return chainEntry.startingBlock;
7103   }
7104 
7105   return BLOCK_END_OF_CHAIN;
7106 }
7107 
7108 /* Read and save the index of all blocks in this stream. */
7109 static HRESULT BlockChainStream_UpdateIndexCache(BlockChainStream* This)
7110 {
7111   ULONG  next_sector, next_offset;
7112   HRESULT hr;
7113   struct BlockChainRun *last_run;
7114 
7115   if (This->indexCacheLen == 0)
7116   {
7117     last_run = NULL;
7118     next_offset = 0;
7119     next_sector = BlockChainStream_GetHeadOfChain(This);
7120   }
7121   else
7122   {
7123     last_run = &This->indexCache[This->indexCacheLen-1];
7124     next_offset = last_run->lastOffset+1;
7125     hr = StorageImpl_GetNextBlockInChain(This->parentStorage,
7126         last_run->firstSector + last_run->lastOffset - last_run->firstOffset,
7127         &next_sector);
7128     if (FAILED(hr)) return hr;
7129   }
7130 
7131   while (next_sector != BLOCK_END_OF_CHAIN)
7132   {
7133     if (!last_run || next_sector != last_run->firstSector + next_offset - last_run->firstOffset)
7134     {
7135       /* Add the current block to the cache. */
7136       if (This->indexCacheSize == 0)
7137       {
7138         This->indexCache = HeapAlloc(GetProcessHeap(), 0, sizeof(struct BlockChainRun)*16);
7139         if (!This->indexCache) return E_OUTOFMEMORY;
7140         This->indexCacheSize = 16;
7141       }
7142       else if (This->indexCacheSize == This->indexCacheLen)
7143       {
7144         struct BlockChainRun *new_cache;
7145         ULONG new_size;
7146 
7147         new_size = This->indexCacheSize * 2;
7148         new_cache = HeapAlloc(GetProcessHeap(), 0, sizeof(struct BlockChainRun)*new_size);
7149         if (!new_cache) return E_OUTOFMEMORY;
7150         memcpy(new_cache, This->indexCache, sizeof(struct BlockChainRun)*This->indexCacheLen);
7151 
7152         HeapFree(GetProcessHeap(), 0, This->indexCache);
7153         This->indexCache = new_cache;
7154         This->indexCacheSize = new_size;
7155       }
7156 
7157       This->indexCacheLen++;
7158       last_run = &This->indexCache[This->indexCacheLen-1];
7159       last_run->firstSector = next_sector;
7160       last_run->firstOffset = next_offset;
7161     }
7162 
7163     last_run->lastOffset = next_offset;
7164 
7165     /* Find the next block. */
7166     next_offset++;
7167     hr = StorageImpl_GetNextBlockInChain(This->parentStorage, next_sector, &next_sector);
7168     if (FAILED(hr)) return hr;
7169   }
7170 
7171   if (This->indexCacheLen)
7172   {
7173     This->tailIndex = last_run->firstSector + last_run->lastOffset - last_run->firstOffset;
7174     This->numBlocks = last_run->lastOffset+1;
7175   }
7176   else
7177   {
7178     This->tailIndex = BLOCK_END_OF_CHAIN;
7179     This->numBlocks = 0;
7180   }
7181 
7182   return S_OK;
7183 }
7184 
7185 /* Locate the nth block in this stream. */
7186 static ULONG BlockChainStream_GetSectorOfOffset(BlockChainStream *This, ULONG offset)
7187 {
7188   ULONG min_offset = 0, max_offset = This->numBlocks-1;
7189   ULONG min_run = 0, max_run = This->indexCacheLen-1;
7190 
7191   if (offset >= This->numBlocks)
7192     return BLOCK_END_OF_CHAIN;
7193 
7194   while (min_run < max_run)
7195   {
7196     ULONG run_to_check = min_run + (offset - min_offset) * (max_run - min_run) / (max_offset - min_offset);
7197     if (offset < This->indexCache[run_to_check].firstOffset)
7198     {
7199       max_offset = This->indexCache[run_to_check].firstOffset-1;
7200       max_run = run_to_check-1;
7201     }
7202     else if (offset > This->indexCache[run_to_check].lastOffset)
7203     {
7204       min_offset = This->indexCache[run_to_check].lastOffset+1;
7205       min_run = run_to_check+1;
7206     }
7207     else
7208       /* Block is in this run. */
7209       min_run = max_run = run_to_check;
7210   }
7211 
7212   return This->indexCache[min_run].firstSector + offset - This->indexCache[min_run].firstOffset;
7213 }
7214 
7215 static HRESULT BlockChainStream_GetBlockAtOffset(BlockChainStream *This,
7216     ULONG index, BlockChainBlock **block, ULONG *sector, BOOL create)
7217 {
7218   BlockChainBlock *result=NULL;
7219   int i;
7220 
7221   for (i=0; i<2; i++)
7222     if (This->cachedBlocks[i].index == index)
7223     {
7224       *sector = This->cachedBlocks[i].sector;
7225       *block = &This->cachedBlocks[i];
7226       return S_OK;
7227     }
7228 
7229   *sector = BlockChainStream_GetSectorOfOffset(This, index);
7230   if (*sector == BLOCK_END_OF_CHAIN)
7231     return STG_E_DOCFILECORRUPT;
7232 
7233   if (create)
7234   {
7235     if (This->cachedBlocks[0].index == 0xffffffff)
7236       result = &This->cachedBlocks[0];
7237     else if (This->cachedBlocks[1].index == 0xffffffff)
7238       result = &This->cachedBlocks[1];
7239     else
7240     {
7241       result = &This->cachedBlocks[This->blockToEvict++];
7242       if (This->blockToEvict == 2)
7243         This->blockToEvict = 0;
7244     }
7245 
7246     if (result->dirty)
7247     {
7248       if (!StorageImpl_WriteBigBlock(This->parentStorage, result->sector, result->data))
7249         return STG_E_WRITEFAULT;
7250       result->dirty = FALSE;
7251     }
7252 
7253     result->read = FALSE;
7254     result->index = index;
7255     result->sector = *sector;
7256   }
7257 
7258   *block = result;
7259   return S_OK;
7260 }
7261 
7262 BlockChainStream* BlockChainStream_Construct(
7263   StorageImpl* parentStorage,
7264   ULONG*         headOfStreamPlaceHolder,
7265   DirRef         dirEntry)
7266 {
7267   BlockChainStream* newStream;
7268 
7269   newStream = HeapAlloc(GetProcessHeap(), 0, sizeof(BlockChainStream));
7270   if(!newStream)
7271     return NULL;
7272 
7273   newStream->parentStorage           = parentStorage;
7274   newStream->headOfStreamPlaceHolder = headOfStreamPlaceHolder;
7275   newStream->ownerDirEntry           = dirEntry;
7276   newStream->indexCache              = NULL;
7277   newStream->indexCacheLen           = 0;
7278   newStream->indexCacheSize          = 0;
7279   newStream->cachedBlocks[0].index = 0xffffffff;
7280   newStream->cachedBlocks[0].dirty = FALSE;
7281   newStream->cachedBlocks[1].index = 0xffffffff;
7282   newStream->cachedBlocks[1].dirty = FALSE;
7283   newStream->blockToEvict          = 0;
7284 
7285   if (FAILED(BlockChainStream_UpdateIndexCache(newStream)))
7286   {
7287     HeapFree(GetProcessHeap(), 0, newStream->indexCache);
7288     HeapFree(GetProcessHeap(), 0, newStream);
7289     return NULL;
7290   }
7291 
7292   return newStream;
7293 }
7294 
7295 HRESULT BlockChainStream_Flush(BlockChainStream* This)
7296 {
7297   int i;
7298   if (!This) return S_OK;
7299   for (i=0; i<2; i++)
7300   {
7301     if (This->cachedBlocks[i].dirty)
7302     {
7303       if (StorageImpl_WriteBigBlock(This->parentStorage, This->cachedBlocks[i].sector, This->cachedBlocks[i].data))
7304         This->cachedBlocks[i].dirty = FALSE;
7305       else
7306         return STG_E_WRITEFAULT;
7307     }
7308   }
7309   return S_OK;
7310 }
7311 
7312 void BlockChainStream_Destroy(BlockChainStream* This)
7313 {
7314   if (This)
7315   {
7316     BlockChainStream_Flush(This);
7317     HeapFree(GetProcessHeap(), 0, This->indexCache);
7318   }
7319   HeapFree(GetProcessHeap(), 0, This);
7320 }
7321 
7322 /******************************************************************************
7323  *      BlockChainStream_Shrink
7324  *
7325  * Shrinks this chain in the big block depot.
7326  */
7327 static BOOL BlockChainStream_Shrink(BlockChainStream* This,
7328                                     ULARGE_INTEGER    newSize)
7329 {
7330   ULONG blockIndex;
7331   ULONG numBlocks;
7332   int i;
7333 
7334   /*
7335    * Figure out how many blocks are needed to contain the new size
7336    */
7337   numBlocks = newSize.QuadPart / This->parentStorage->bigBlockSize;
7338 
7339   if ((newSize.QuadPart % This->parentStorage->bigBlockSize) != 0)
7340     numBlocks++;
7341 
7342   if (numBlocks)
7343   {
7344     /*
7345      * Go to the new end of chain
7346      */
7347     blockIndex = BlockChainStream_GetSectorOfOffset(This, numBlocks-1);
7348 
7349     /* Mark the new end of chain */
7350     StorageImpl_SetNextBlockInChain(
7351       This->parentStorage,
7352       blockIndex,
7353       BLOCK_END_OF_CHAIN);
7354 
7355     This->tailIndex = blockIndex;
7356   }
7357   else
7358   {
7359     if (This->headOfStreamPlaceHolder != 0)
7360     {
7361       *This->headOfStreamPlaceHolder = BLOCK_END_OF_CHAIN;
7362     }
7363     else
7364     {
7365       DirEntry chainEntry;
7366       assert(This->ownerDirEntry != DIRENTRY_NULL);
7367 
7368       StorageImpl_ReadDirEntry(
7369         This->parentStorage,
7370         This->ownerDirEntry,
7371         &chainEntry);
7372 
7373       chainEntry.startingBlock = BLOCK_END_OF_CHAIN;
7374 
7375       StorageImpl_WriteDirEntry(
7376         This->parentStorage,
7377         This->ownerDirEntry,
7378         &chainEntry);
7379     }
7380 
7381     This->tailIndex = BLOCK_END_OF_CHAIN;
7382   }
7383 
7384   This->numBlocks = numBlocks;
7385 
7386   /*
7387    * Mark the extra blocks as free
7388    */
7389   while (This->indexCacheLen && This->indexCache[This->indexCacheLen-1].lastOffset >= numBlocks)
7390   {
7391     struct BlockChainRun *last_run = &This->indexCache[This->indexCacheLen-1];
7392     StorageImpl_FreeBigBlock(This->parentStorage,
7393       last_run->firstSector + last_run->lastOffset - last_run->firstOffset);
7394     if (last_run->lastOffset == last_run->firstOffset)
7395       This->indexCacheLen--;
7396     else
7397       last_run->lastOffset--;
7398   }
7399 
7400   /*
7401    * Reset the last accessed block cache.
7402    */
7403   for (i=0; i<2; i++)
7404   {
7405     if (This->cachedBlocks[i].index >= numBlocks)
7406     {
7407       This->cachedBlocks[i].index = 0xffffffff;
7408       This->cachedBlocks[i].dirty = FALSE;
7409     }
7410   }
7411 
7412   return TRUE;
7413 }
7414 
7415 /******************************************************************************
7416  *      BlockChainStream_Enlarge
7417  *
7418  * Grows this chain in the big block depot.
7419  */
7420 static BOOL BlockChainStream_Enlarge(BlockChainStream* This,
7421                                      ULARGE_INTEGER    newSize)
7422 {
7423   ULONG blockIndex, currentBlock;
7424   ULONG newNumBlocks;
7425   ULONG oldNumBlocks = 0;
7426 
7427   blockIndex = BlockChainStream_GetHeadOfChain(This);
7428 
7429   /*
7430    * Empty chain. Create the head.
7431    */
7432   if (blockIndex == BLOCK_END_OF_CHAIN)
7433   {
7434     blockIndex = StorageImpl_GetNextFreeBigBlock(This->parentStorage);
7435     StorageImpl_SetNextBlockInChain(This->parentStorage,
7436                                       blockIndex,
7437                                       BLOCK_END_OF_CHAIN);
7438 
7439     if (This->headOfStreamPlaceHolder != 0)
7440     {
7441       *(This->headOfStreamPlaceHolder) = blockIndex;
7442     }
7443     else
7444     {
7445       DirEntry chainEntry;
7446       assert(This->ownerDirEntry != DIRENTRY_NULL);
7447 
7448       StorageImpl_ReadDirEntry(
7449         This->parentStorage,
7450         This->ownerDirEntry,
7451         &chainEntry);
7452 
7453       chainEntry.startingBlock = blockIndex;
7454 
7455       StorageImpl_WriteDirEntry(
7456         This->parentStorage,
7457         This->ownerDirEntry,
7458         &chainEntry);
7459     }
7460 
7461     This->tailIndex = blockIndex;
7462     This->numBlocks = 1;
7463   }
7464 
7465   /*
7466    * Figure out how many blocks are needed to contain this stream
7467    */
7468   newNumBlocks = newSize.QuadPart / This->parentStorage->bigBlockSize;
7469 
7470   if ((newSize.QuadPart % This->parentStorage->bigBlockSize) != 0)
7471     newNumBlocks++;
7472 
7473   /*
7474    * Go to the current end of chain
7475    */
7476   if (This->tailIndex == BLOCK_END_OF_CHAIN)
7477   {
7478     currentBlock = blockIndex;
7479 
7480     while (blockIndex != BLOCK_END_OF_CHAIN)
7481     {
7482       This->numBlocks++;
7483       currentBlock = blockIndex;
7484 
7485       if(FAILED(StorageImpl_GetNextBlockInChain(This->parentStorage, currentBlock,
7486 						&blockIndex)))
7487 	return FALSE;
7488     }
7489 
7490     This->tailIndex = currentBlock;
7491   }
7492 
7493   currentBlock = This->tailIndex;
7494   oldNumBlocks = This->numBlocks;
7495 
7496   /*
7497    * Add new blocks to the chain
7498    */
7499   if (oldNumBlocks < newNumBlocks)
7500   {
7501     while (oldNumBlocks < newNumBlocks)
7502     {
7503       blockIndex = StorageImpl_GetNextFreeBigBlock(This->parentStorage);
7504 
7505       StorageImpl_SetNextBlockInChain(
7506 	This->parentStorage,
7507 	currentBlock,
7508 	blockIndex);
7509 
7510       StorageImpl_SetNextBlockInChain(
7511         This->parentStorage,
7512 	blockIndex,
7513 	BLOCK_END_OF_CHAIN);
7514 
7515       currentBlock = blockIndex;
7516       oldNumBlocks++;
7517     }
7518 
7519     This->tailIndex = blockIndex;
7520     This->numBlocks = newNumBlocks;
7521   }
7522 
7523   if (FAILED(BlockChainStream_UpdateIndexCache(This)))
7524     return FALSE;
7525 
7526   return TRUE;
7527 }
7528 
7529 
7530 /******************************************************************************
7531  *      BlockChainStream_GetSize
7532  *
7533  * Returns the size of this chain.
7534  * Will return the block count if this chain doesn't have a directory entry.
7535  */
7536 static ULARGE_INTEGER BlockChainStream_GetSize(BlockChainStream* This)
7537 {
7538   DirEntry chainEntry;
7539 
7540   if(This->headOfStreamPlaceHolder == NULL)
7541   {
7542     /*
7543      * This chain has a directory entry so use the size value from there.
7544      */
7545     StorageImpl_ReadDirEntry(
7546       This->parentStorage,
7547       This->ownerDirEntry,
7548       &chainEntry);
7549 
7550     return chainEntry.size;
7551   }
7552   else
7553   {
7554     /*
7555      * this chain is a chain that does not have a directory entry, figure out the
7556      * size by making the product number of used blocks times the
7557      * size of them
7558      */
7559     ULARGE_INTEGER result;
7560     result.QuadPart =
7561       (ULONGLONG)BlockChainStream_GetCount(This) *
7562       This->parentStorage->bigBlockSize;
7563 
7564     return result;
7565   }
7566 }
7567 
7568 /******************************************************************************
7569  *      BlockChainStream_SetSize
7570  *
7571  * Sets the size of this stream. The big block depot will be updated.
7572  * The file will grow if we grow the chain.
7573  *
7574  * TODO: Free the actual blocks in the file when we shrink the chain.
7575  *       Currently, the blocks are still in the file. So the file size
7576  *       doesn't shrink even if we shrink streams.
7577  */
7578 BOOL BlockChainStream_SetSize(
7579   BlockChainStream* This,
7580   ULARGE_INTEGER    newSize)
7581 {
7582   ULARGE_INTEGER size = BlockChainStream_GetSize(This);
7583 
7584   if (newSize.QuadPart == size.QuadPart)
7585     return TRUE;
7586 
7587   if (newSize.QuadPart < size.QuadPart)
7588   {
7589     BlockChainStream_Shrink(This, newSize);
7590   }
7591   else
7592   {
7593     BlockChainStream_Enlarge(This, newSize);
7594   }
7595 
7596   return TRUE;
7597 }
7598 
7599 /******************************************************************************
7600  *      BlockChainStream_ReadAt
7601  *
7602  * Reads a specified number of bytes from this chain at the specified offset.
7603  * bytesRead may be NULL.
7604  * Failure will be returned if the specified number of bytes has not been read.
7605  */
7606 HRESULT BlockChainStream_ReadAt(BlockChainStream* This,
7607   ULARGE_INTEGER offset,
7608   ULONG          size,
7609   void*          buffer,
7610   ULONG*         bytesRead)
7611 {
7612   ULONG blockNoInSequence = offset.QuadPart / This->parentStorage->bigBlockSize;
7613   ULONG offsetInBlock     = offset.QuadPart % This->parentStorage->bigBlockSize;
7614   ULONG bytesToReadInBuffer;
7615   ULONG blockIndex;
7616   BYTE* bufferWalker;
7617   ULARGE_INTEGER stream_size;
7618   HRESULT hr;
7619   BlockChainBlock *cachedBlock;
7620 
7621   TRACE("(%p)-> %i %p %i %p\n",This, offset.u.LowPart, buffer, size, bytesRead);
7622 
7623   /*
7624    * Find the first block in the stream that contains part of the buffer.
7625    */
7626   blockIndex = BlockChainStream_GetSectorOfOffset(This, blockNoInSequence);
7627 
7628   *bytesRead   = 0;
7629 
7630   stream_size = BlockChainStream_GetSize(This);
7631   if (stream_size.QuadPart > offset.QuadPart)
7632     size = min(stream_size.QuadPart - offset.QuadPart, size);
7633   else
7634     return S_OK;
7635 
7636   /*
7637    * Start reading the buffer.
7638    */
7639   bufferWalker = buffer;
7640 
7641   while (size > 0)
7642   {
7643     ULARGE_INTEGER ulOffset;
7644     DWORD bytesReadAt;
7645 
7646     /*
7647      * Calculate how many bytes we can copy from this big block.
7648      */
7649     bytesToReadInBuffer =
7650       min(This->parentStorage->bigBlockSize - offsetInBlock, size);
7651 
7652     hr = BlockChainStream_GetBlockAtOffset(This, blockNoInSequence, &cachedBlock, &blockIndex, size == bytesToReadInBuffer);
7653 
7654     if (FAILED(hr))
7655       return hr;
7656 
7657     if (!cachedBlock)
7658     {
7659       /* Not in cache, and we're going to read past the end of the block. */
7660       ulOffset.QuadPart = StorageImpl_GetBigBlockOffset(This->parentStorage, blockIndex) +
7661                                offsetInBlock;
7662 
7663       StorageImpl_ReadAt(This->parentStorage,
7664            ulOffset,
7665            bufferWalker,
7666            bytesToReadInBuffer,
7667            &bytesReadAt);
7668     }
7669     else
7670     {
7671       if (!cachedBlock->read)
7672       {
7673         ULONG read;
7674         if (FAILED(StorageImpl_ReadBigBlock(This->parentStorage, cachedBlock->sector, cachedBlock->data, &read)) && !read)
7675           return STG_E_READFAULT;
7676 
7677         cachedBlock->read = TRUE;
7678       }
7679 
7680       memcpy(bufferWalker, cachedBlock->data+offsetInBlock, bytesToReadInBuffer);
7681       bytesReadAt = bytesToReadInBuffer;
7682     }
7683 
7684     blockNoInSequence++;
7685     bufferWalker += bytesReadAt;
7686     size         -= bytesReadAt;
7687     *bytesRead   += bytesReadAt;
7688     offsetInBlock = 0;  /* There is no offset on the next block */
7689 
7690     if (bytesToReadInBuffer != bytesReadAt)
7691         break;
7692   }
7693 
7694   return S_OK;
7695 }
7696 
7697 /******************************************************************************
7698  *      BlockChainStream_WriteAt
7699  *
7700  * Writes the specified number of bytes to this chain at the specified offset.
7701  * Will fail if not all specified number of bytes have been written.
7702  */
7703 HRESULT BlockChainStream_WriteAt(BlockChainStream* This,
7704   ULARGE_INTEGER    offset,
7705   ULONG             size,
7706   const void*       buffer,
7707   ULONG*            bytesWritten)
7708 {
7709   ULONG blockNoInSequence = offset.QuadPart / This->parentStorage->bigBlockSize;
7710   ULONG offsetInBlock     = offset.QuadPart % This->parentStorage->bigBlockSize;
7711   ULONG bytesToWrite;
7712   ULONG blockIndex;
7713   const BYTE* bufferWalker;
7714   HRESULT hr;
7715   BlockChainBlock *cachedBlock;
7716 
7717   *bytesWritten   = 0;
7718   bufferWalker = buffer;
7719 
7720   while (size > 0)
7721   {
7722     ULARGE_INTEGER ulOffset;
7723     DWORD bytesWrittenAt;
7724 
7725     /*
7726      * Calculate how many bytes we can copy to this big block.
7727      */
7728     bytesToWrite =
7729       min(This->parentStorage->bigBlockSize - offsetInBlock, size);
7730 
7731     hr = BlockChainStream_GetBlockAtOffset(This, blockNoInSequence, &cachedBlock, &blockIndex, size == bytesToWrite);
7732 
7733     /* BlockChainStream_SetSize should have already been called to ensure we have
7734      * enough blocks in the chain to write into */
7735     if (FAILED(hr))
7736     {
7737       ERR("not enough blocks in chain to write data\n");
7738       return hr;
7739     }
7740 
7741     if (!cachedBlock)
7742     {
7743       /* Not in cache, and we're going to write past the end of the block. */
7744       ulOffset.QuadPart = StorageImpl_GetBigBlockOffset(This->parentStorage, blockIndex) +
7745                                offsetInBlock;
7746 
7747       StorageImpl_WriteAt(This->parentStorage,
7748            ulOffset,
7749            bufferWalker,
7750            bytesToWrite,
7751            &bytesWrittenAt);
7752     }
7753     else
7754     {
7755       if (!cachedBlock->read && bytesToWrite != This->parentStorage->bigBlockSize)
7756       {
7757         ULONG read;
7758         if (FAILED(StorageImpl_ReadBigBlock(This->parentStorage, cachedBlock->sector, cachedBlock->data, &read)) && !read)
7759           return STG_E_READFAULT;
7760       }
7761 
7762       memcpy(cachedBlock->data+offsetInBlock, bufferWalker, bytesToWrite);
7763       bytesWrittenAt = bytesToWrite;
7764       cachedBlock->read = TRUE;
7765       cachedBlock->dirty = TRUE;
7766     }
7767 
7768     blockNoInSequence++;
7769     bufferWalker  += bytesWrittenAt;
7770     size          -= bytesWrittenAt;
7771     *bytesWritten += bytesWrittenAt;
7772     offsetInBlock  = 0;      /* There is no offset on the next block */
7773 
7774     if (bytesWrittenAt != bytesToWrite)
7775       break;
7776   }
7777 
7778   return (size == 0) ? S_OK : STG_E_WRITEFAULT;
7779 }
7780 
7781 
7782 /************************************************************************
7783  * SmallBlockChainStream implementation
7784  ***********************************************************************/
7785 
7786 SmallBlockChainStream* SmallBlockChainStream_Construct(
7787   StorageImpl* parentStorage,
7788   ULONG*         headOfStreamPlaceHolder,
7789   DirRef         dirEntry)
7790 {
7791   SmallBlockChainStream* newStream;
7792 
7793   newStream = HeapAlloc(GetProcessHeap(), 0, sizeof(SmallBlockChainStream));
7794 
7795   newStream->parentStorage      = parentStorage;
7796   newStream->headOfStreamPlaceHolder = headOfStreamPlaceHolder;
7797   newStream->ownerDirEntry      = dirEntry;
7798 
7799   return newStream;
7800 }
7801 
7802 void SmallBlockChainStream_Destroy(
7803   SmallBlockChainStream* This)
7804 {
7805   HeapFree(GetProcessHeap(), 0, This);
7806 }
7807 
7808 /******************************************************************************
7809  *      SmallBlockChainStream_GetHeadOfChain
7810  *
7811  * Returns the head of this chain of small blocks.
7812  */
7813 static ULONG SmallBlockChainStream_GetHeadOfChain(
7814   SmallBlockChainStream* This)
7815 {
7816   DirEntry  chainEntry;
7817   HRESULT   hr;
7818 
7819   if (This->headOfStreamPlaceHolder != NULL)
7820     return *(This->headOfStreamPlaceHolder);
7821 
7822   if (This->ownerDirEntry)
7823   {
7824     hr = StorageImpl_ReadDirEntry(
7825                       This->parentStorage,
7826                       This->ownerDirEntry,
7827                       &chainEntry);
7828 
7829     if (SUCCEEDED(hr) && chainEntry.startingBlock < BLOCK_FIRST_SPECIAL)
7830       return chainEntry.startingBlock;
7831   }
7832 
7833   return BLOCK_END_OF_CHAIN;
7834 }
7835 
7836 /******************************************************************************
7837  *      SmallBlockChainStream_GetNextBlockInChain
7838  *
7839  * Returns the index of the next small block in this chain.
7840  *
7841  * Return Values:
7842  *    - BLOCK_END_OF_CHAIN: end of this chain
7843  *    - BLOCK_UNUSED: small block 'blockIndex' is free
7844  */
7845 static HRESULT SmallBlockChainStream_GetNextBlockInChain(
7846   SmallBlockChainStream* This,
7847   ULONG                  blockIndex,
7848   ULONG*                 nextBlockInChain)
7849 {
7850   ULARGE_INTEGER offsetOfBlockInDepot;
7851   DWORD  buffer;
7852   ULONG  bytesRead;
7853   HRESULT res;
7854 
7855   *nextBlockInChain = BLOCK_END_OF_CHAIN;
7856 
7857   offsetOfBlockInDepot.QuadPart  = (ULONGLONG)blockIndex * sizeof(ULONG);
7858 
7859   /*
7860    * Read those bytes in the buffer from the small block file.
7861    */
7862   res = BlockChainStream_ReadAt(
7863               This->parentStorage->smallBlockDepotChain,
7864               offsetOfBlockInDepot,
7865               sizeof(DWORD),
7866               &buffer,
7867               &bytesRead);
7868 
7869   if (SUCCEEDED(res) && bytesRead != sizeof(DWORD))
7870     res = STG_E_READFAULT;
7871 
7872   if (SUCCEEDED(res))
7873   {
7874     StorageUtl_ReadDWord((BYTE *)&buffer, 0, nextBlockInChain);
7875     return S_OK;
7876   }
7877 
7878   return res;
7879 }
7880 
7881 /******************************************************************************
7882  *       SmallBlockChainStream_SetNextBlockInChain
7883  *
7884  * Writes the index of the next block of the specified block in the small
7885  * block depot.
7886  * To set the end of chain use BLOCK_END_OF_CHAIN as nextBlock.
7887  * To flag a block as free use BLOCK_UNUSED as nextBlock.
7888  */
7889 static void SmallBlockChainStream_SetNextBlockInChain(
7890   SmallBlockChainStream* This,
7891   ULONG                  blockIndex,
7892   ULONG                  nextBlock)
7893 {
7894   ULARGE_INTEGER offsetOfBlockInDepot;
7895   DWORD  buffer;
7896   ULONG  bytesWritten;
7897 
7898   offsetOfBlockInDepot.QuadPart  = (ULONGLONG)blockIndex * sizeof(ULONG);
7899 
7900   StorageUtl_WriteDWord((BYTE *)&buffer, 0, nextBlock);
7901 
7902   /*
7903    * Read those bytes in the buffer from the small block file.
7904    */
7905   BlockChainStream_WriteAt(
7906     This->parentStorage->smallBlockDepotChain,
7907     offsetOfBlockInDepot,
7908     sizeof(DWORD),
7909     &buffer,
7910     &bytesWritten);
7911 }
7912 
7913 /******************************************************************************
7914  *      SmallBlockChainStream_FreeBlock
7915  *
7916  * Flag small block 'blockIndex' as free in the small block depot.
7917  */
7918 static void SmallBlockChainStream_FreeBlock(
7919   SmallBlockChainStream* This,
7920   ULONG                  blockIndex)
7921 {
7922   SmallBlockChainStream_SetNextBlockInChain(This, blockIndex, BLOCK_UNUSED);
7923 }
7924 
7925 /******************************************************************************
7926  *      SmallBlockChainStream_GetNextFreeBlock
7927  *
7928  * Returns the index of a free small block. The small block depot will be
7929  * enlarged if necessary. The small block chain will also be enlarged if
7930  * necessary.
7931  */
7932 static ULONG SmallBlockChainStream_GetNextFreeBlock(
7933   SmallBlockChainStream* This)
7934 {
7935   ULARGE_INTEGER offsetOfBlockInDepot;
7936   DWORD buffer;
7937   ULONG bytesRead;
7938   ULONG blockIndex = This->parentStorage->firstFreeSmallBlock;
7939   ULONG nextBlockIndex = BLOCK_END_OF_CHAIN;
7940   HRESULT res = S_OK;
7941   ULONG smallBlocksPerBigBlock;
7942   DirEntry rootEntry;
7943   ULONG blocksRequired;
7944   ULARGE_INTEGER old_size, size_required;
7945 
7946   offsetOfBlockInDepot.u.HighPart = 0;
7947 
7948   /*
7949    * Scan the small block depot for a free block
7950    */
7951   while (nextBlockIndex != BLOCK_UNUSED)
7952   {
7953     offsetOfBlockInDepot.QuadPart = (ULONGLONG)blockIndex * sizeof(ULONG);
7954 
7955     res = BlockChainStream_ReadAt(
7956                 This->parentStorage->smallBlockDepotChain,
7957                 offsetOfBlockInDepot,
7958                 sizeof(DWORD),
7959                 &buffer,
7960                 &bytesRead);
7961 
7962     /*
7963      * If we run out of space for the small block depot, enlarge it
7964      */
7965     if (SUCCEEDED(res) && bytesRead == sizeof(DWORD))
7966     {
7967       StorageUtl_ReadDWord((BYTE *)&buffer, 0, &nextBlockIndex);
7968 
7969       if (nextBlockIndex != BLOCK_UNUSED)
7970         blockIndex++;
7971     }
7972     else
7973     {
7974       ULONG count =
7975         BlockChainStream_GetCount(This->parentStorage->smallBlockDepotChain);
7976 
7977       BYTE smallBlockDepot[MAX_BIG_BLOCK_SIZE];
7978       ULARGE_INTEGER newSize, offset;
7979       ULONG bytesWritten;
7980 
7981       newSize.QuadPart = (ULONGLONG)(count + 1) * This->parentStorage->bigBlockSize;
7982       BlockChainStream_Enlarge(This->parentStorage->smallBlockDepotChain, newSize);
7983 
7984       /*
7985        * Initialize all the small blocks to free
7986        */
7987       memset(smallBlockDepot, BLOCK_UNUSED, This->parentStorage->bigBlockSize);
7988       offset.QuadPart = (ULONGLONG)count * This->parentStorage->bigBlockSize;
7989       BlockChainStream_WriteAt(This->parentStorage->smallBlockDepotChain,
7990         offset, This->parentStorage->bigBlockSize, smallBlockDepot, &bytesWritten);
7991 
7992       StorageImpl_SaveFileHeader(This->parentStorage);
7993     }
7994   }
7995 
7996   This->parentStorage->firstFreeSmallBlock = blockIndex+1;
7997 
7998   smallBlocksPerBigBlock =
7999     This->parentStorage->bigBlockSize / This->parentStorage->smallBlockSize;
8000 
8001   /*
8002    * Verify if we have to allocate big blocks to contain small blocks
8003    */
8004   blocksRequired = (blockIndex / smallBlocksPerBigBlock) + 1;
8005 
8006   size_required.QuadPart = (ULONGLONG)blocksRequired * This->parentStorage->bigBlockSize;
8007 
8008   old_size = BlockChainStream_GetSize(This->parentStorage->smallBlockRootChain);
8009 
8010   if (size_required.QuadPart > old_size.QuadPart)
8011   {
8012     BlockChainStream_SetSize(
8013       This->parentStorage->smallBlockRootChain,
8014       size_required);
8015 
8016     StorageImpl_ReadDirEntry(
8017       This->parentStorage,
8018       This->parentStorage->base.storageDirEntry,
8019       &rootEntry);
8020 
8021     rootEntry.size = size_required;
8022 
8023     StorageImpl_WriteDirEntry(
8024       This->parentStorage,
8025       This->parentStorage->base.storageDirEntry,
8026       &rootEntry);
8027   }
8028 
8029   return blockIndex;
8030 }
8031 
8032 /******************************************************************************
8033  *      SmallBlockChainStream_ReadAt
8034  *
8035  * Reads a specified number of bytes from this chain at the specified offset.
8036  * bytesRead may be NULL.
8037  * Failure will be returned if the specified number of bytes has not been read.
8038  */
8039 HRESULT SmallBlockChainStream_ReadAt(
8040   SmallBlockChainStream* This,
8041   ULARGE_INTEGER         offset,
8042   ULONG                  size,
8043   void*                  buffer,
8044   ULONG*                 bytesRead)
8045 {
8046   HRESULT rc = S_OK;
8047   ULARGE_INTEGER offsetInBigBlockFile;
8048   ULONG blockNoInSequence =
8049     offset.u.LowPart / This->parentStorage->smallBlockSize;
8050 
8051   ULONG offsetInBlock = offset.u.LowPart % This->parentStorage->smallBlockSize;
8052   ULONG bytesToReadInBuffer;
8053   ULONG blockIndex;
8054   ULONG bytesReadFromBigBlockFile;
8055   BYTE* bufferWalker;
8056   ULARGE_INTEGER stream_size;
8057 
8058   /*
8059    * This should never happen on a small block file.
8060    */
8061   assert(offset.u.HighPart==0);
8062 
8063   *bytesRead   = 0;
8064 
8065   stream_size = SmallBlockChainStream_GetSize(This);
8066   if (stream_size.QuadPart > offset.QuadPart)
8067     size = min(stream_size.QuadPart - offset.QuadPart, size);
8068   else
8069     return S_OK;
8070 
8071   /*
8072    * Find the first block in the stream that contains part of the buffer.
8073    */
8074   blockIndex = SmallBlockChainStream_GetHeadOfChain(This);
8075 
8076   while ( (blockNoInSequence > 0) &&  (blockIndex != BLOCK_END_OF_CHAIN))
8077   {
8078     rc = SmallBlockChainStream_GetNextBlockInChain(This, blockIndex, &blockIndex);
8079     if(FAILED(rc))
8080       return rc;
8081     blockNoInSequence--;
8082   }
8083 
8084   /*
8085    * Start reading the buffer.
8086    */
8087   bufferWalker = buffer;
8088 
8089   while ( (size > 0) && (blockIndex != BLOCK_END_OF_CHAIN) )
8090   {
8091     /*
8092      * Calculate how many bytes we can copy from this small block.
8093      */
8094     bytesToReadInBuffer =
8095       min(This->parentStorage->smallBlockSize - offsetInBlock, size);
8096 
8097     /*
8098      * Calculate the offset of the small block in the small block file.
8099      */
8100     offsetInBigBlockFile.QuadPart   =
8101       (ULONGLONG)blockIndex * This->parentStorage->smallBlockSize;
8102 
8103     offsetInBigBlockFile.QuadPart  += offsetInBlock;
8104 
8105     /*
8106      * Read those bytes in the buffer from the small block file.
8107      * The small block has already been identified so it shouldn't fail
8108      * unless the file is corrupt.
8109      */
8110     rc = BlockChainStream_ReadAt(This->parentStorage->smallBlockRootChain,
8111       offsetInBigBlockFile,
8112       bytesToReadInBuffer,
8113       bufferWalker,
8114       &bytesReadFromBigBlockFile);
8115 
8116     if (FAILED(rc))
8117       return rc;
8118 
8119     if (!bytesReadFromBigBlockFile)
8120       return STG_E_DOCFILECORRUPT;
8121 
8122     /*
8123      * Step to the next big block.
8124      */
8125     rc = SmallBlockChainStream_GetNextBlockInChain(This, blockIndex, &blockIndex);
8126     if(FAILED(rc))
8127       return STG_E_DOCFILECORRUPT;
8128 
8129     bufferWalker += bytesReadFromBigBlockFile;
8130     size         -= bytesReadFromBigBlockFile;
8131     *bytesRead   += bytesReadFromBigBlockFile;
8132     offsetInBlock = (offsetInBlock + bytesReadFromBigBlockFile) % This->parentStorage->smallBlockSize;
8133   }
8134 
8135   return S_OK;
8136 }
8137 
8138 /******************************************************************************
8139  *       SmallBlockChainStream_WriteAt
8140  *
8141  * Writes the specified number of bytes to this chain at the specified offset.
8142  * Will fail if not all specified number of bytes have been written.
8143  */
8144 HRESULT SmallBlockChainStream_WriteAt(
8145   SmallBlockChainStream* This,
8146   ULARGE_INTEGER offset,
8147   ULONG          size,
8148   const void*    buffer,
8149   ULONG*         bytesWritten)
8150 {
8151   ULARGE_INTEGER offsetInBigBlockFile;
8152   ULONG blockNoInSequence =
8153     offset.u.LowPart / This->parentStorage->smallBlockSize;
8154 
8155   ULONG offsetInBlock = offset.u.LowPart % This->parentStorage->smallBlockSize;
8156   ULONG bytesToWriteInBuffer;
8157   ULONG blockIndex;
8158   ULONG bytesWrittenToBigBlockFile;
8159   const BYTE* bufferWalker;
8160   HRESULT res;
8161 
8162   /*
8163    * This should never happen on a small block file.
8164    */
8165   assert(offset.u.HighPart==0);
8166 
8167   /*
8168    * Find the first block in the stream that contains part of the buffer.
8169    */
8170   blockIndex = SmallBlockChainStream_GetHeadOfChain(This);
8171 
8172   while ( (blockNoInSequence > 0) &&  (blockIndex != BLOCK_END_OF_CHAIN))
8173   {
8174     if(FAILED(SmallBlockChainStream_GetNextBlockInChain(This, blockIndex, &blockIndex)))
8175       return STG_E_DOCFILECORRUPT;
8176     blockNoInSequence--;
8177   }
8178 
8179   /*
8180    * Start writing the buffer.
8181    */
8182   *bytesWritten   = 0;
8183   bufferWalker = buffer;
8184   while ( (size > 0) && (blockIndex != BLOCK_END_OF_CHAIN) )
8185   {
8186     /*
8187      * Calculate how many bytes we can copy to this small block.
8188      */
8189     bytesToWriteInBuffer =
8190       min(This->parentStorage->smallBlockSize - offsetInBlock, size);
8191 
8192     /*
8193      * Calculate the offset of the small block in the small block file.
8194      */
8195     offsetInBigBlockFile.QuadPart   =
8196       (ULONGLONG)blockIndex * This->parentStorage->smallBlockSize;
8197 
8198     offsetInBigBlockFile.QuadPart  += offsetInBlock;
8199 
8200     /*
8201      * Write those bytes in the buffer to the small block file.
8202      */
8203     res = BlockChainStream_WriteAt(
8204       This->parentStorage->smallBlockRootChain,
8205       offsetInBigBlockFile,
8206       bytesToWriteInBuffer,
8207       bufferWalker,
8208       &bytesWrittenToBigBlockFile);
8209     if (FAILED(res))
8210       return res;
8211 
8212     /*
8213      * Step to the next big block.
8214      */
8215     res = SmallBlockChainStream_GetNextBlockInChain(This, blockIndex, &blockIndex);
8216     if (FAILED(res))
8217       return res;
8218     bufferWalker  += bytesWrittenToBigBlockFile;
8219     size          -= bytesWrittenToBigBlockFile;
8220     *bytesWritten += bytesWrittenToBigBlockFile;
8221     offsetInBlock  = (offsetInBlock + bytesWrittenToBigBlockFile) % This->parentStorage->smallBlockSize;
8222   }
8223 
8224   return (size == 0) ? S_OK : STG_E_WRITEFAULT;
8225 }
8226 
8227 /******************************************************************************
8228  *       SmallBlockChainStream_Shrink
8229  *
8230  * Shrinks this chain in the small block depot.
8231  */
8232 static BOOL SmallBlockChainStream_Shrink(
8233   SmallBlockChainStream* This,
8234   ULARGE_INTEGER newSize)
8235 {
8236   ULONG blockIndex, extraBlock;
8237   ULONG numBlocks;
8238   ULONG count = 0;
8239 
8240   numBlocks = newSize.u.LowPart / This->parentStorage->smallBlockSize;
8241 
8242   if ((newSize.u.LowPart % This->parentStorage->smallBlockSize) != 0)
8243     numBlocks++;
8244 
8245   blockIndex = SmallBlockChainStream_GetHeadOfChain(This);
8246 
8247   /*
8248    * Go to the new end of chain
8249    */
8250   while (count < numBlocks)
8251   {
8252     if(FAILED(SmallBlockChainStream_GetNextBlockInChain(This, blockIndex,
8253 							&blockIndex)))
8254       return FALSE;
8255     count++;
8256   }
8257 
8258   /*
8259    * If the count is 0, we have a special case, the head of the chain was
8260    * just freed.
8261    */
8262   if (count == 0)
8263   {
8264     DirEntry chainEntry;
8265 
8266     StorageImpl_ReadDirEntry(This->parentStorage,
8267 			     This->ownerDirEntry,
8268 			     &chainEntry);
8269 
8270     chainEntry.startingBlock = BLOCK_END_OF_CHAIN;
8271 
8272     StorageImpl_WriteDirEntry(This->parentStorage,
8273 			      This->ownerDirEntry,
8274 			      &chainEntry);
8275 
8276     /*
8277      * We start freeing the chain at the head block.
8278      */
8279     extraBlock = blockIndex;
8280   }
8281   else
8282   {
8283     /* Get the next block before marking the new end */
8284     if(FAILED(SmallBlockChainStream_GetNextBlockInChain(This, blockIndex,
8285 							&extraBlock)))
8286       return FALSE;
8287 
8288     /* Mark the new end of chain */
8289     SmallBlockChainStream_SetNextBlockInChain(
8290       This,
8291       blockIndex,
8292       BLOCK_END_OF_CHAIN);
8293   }
8294 
8295   /*
8296    * Mark the extra blocks as free
8297    */
8298   while (extraBlock != BLOCK_END_OF_CHAIN)
8299   {
8300     if(FAILED(SmallBlockChainStream_GetNextBlockInChain(This, extraBlock,
8301 							&blockIndex)))
8302       return FALSE;
8303     SmallBlockChainStream_FreeBlock(This, extraBlock);
8304     This->parentStorage->firstFreeSmallBlock = min(This->parentStorage->firstFreeSmallBlock, extraBlock);
8305     extraBlock = blockIndex;
8306   }
8307 
8308   return TRUE;
8309 }
8310 
8311 /******************************************************************************
8312  *      SmallBlockChainStream_Enlarge
8313  *
8314  * Grows this chain in the small block depot.
8315  */
8316 static BOOL SmallBlockChainStream_Enlarge(
8317   SmallBlockChainStream* This,
8318   ULARGE_INTEGER newSize)
8319 {
8320   ULONG blockIndex, currentBlock;
8321   ULONG newNumBlocks;
8322   ULONG oldNumBlocks = 0;
8323 
8324   blockIndex = SmallBlockChainStream_GetHeadOfChain(This);
8325 
8326   /*
8327    * Empty chain. Create the head.
8328    */
8329   if (blockIndex == BLOCK_END_OF_CHAIN)
8330   {
8331     blockIndex = SmallBlockChainStream_GetNextFreeBlock(This);
8332     SmallBlockChainStream_SetNextBlockInChain(
8333         This,
8334         blockIndex,
8335         BLOCK_END_OF_CHAIN);
8336 
8337     if (This->headOfStreamPlaceHolder != NULL)
8338     {
8339       *(This->headOfStreamPlaceHolder) = blockIndex;
8340     }
8341     else
8342     {
8343       DirEntry chainEntry;
8344 
8345       StorageImpl_ReadDirEntry(This->parentStorage, This->ownerDirEntry,
8346                                    &chainEntry);
8347 
8348       chainEntry.startingBlock = blockIndex;
8349 
8350       StorageImpl_WriteDirEntry(This->parentStorage, This->ownerDirEntry,
8351                                   &chainEntry);
8352     }
8353   }
8354 
8355   currentBlock = blockIndex;
8356 
8357   /*
8358    * Figure out how many blocks are needed to contain this stream
8359    */
8360   newNumBlocks = newSize.u.LowPart / This->parentStorage->smallBlockSize;
8361 
8362   if ((newSize.u.LowPart % This->parentStorage->smallBlockSize) != 0)
8363     newNumBlocks++;
8364 
8365   /*
8366    * Go to the current end of chain
8367    */
8368   while (blockIndex != BLOCK_END_OF_CHAIN)
8369   {
8370     oldNumBlocks++;
8371     currentBlock = blockIndex;
8372     if(FAILED(SmallBlockChainStream_GetNextBlockInChain(This, currentBlock, &blockIndex)))
8373       return FALSE;
8374   }
8375 
8376   /*
8377    * Add new blocks to the chain
8378    */
8379   while (oldNumBlocks < newNumBlocks)
8380   {
8381     blockIndex = SmallBlockChainStream_GetNextFreeBlock(This);
8382     SmallBlockChainStream_SetNextBlockInChain(This, currentBlock, blockIndex);
8383 
8384     SmallBlockChainStream_SetNextBlockInChain(
8385       This,
8386       blockIndex,
8387       BLOCK_END_OF_CHAIN);
8388 
8389     currentBlock = blockIndex;
8390     oldNumBlocks++;
8391   }
8392 
8393   return TRUE;
8394 }
8395 
8396 /******************************************************************************
8397  *      SmallBlockChainStream_SetSize
8398  *
8399  * Sets the size of this stream.
8400  * The file will grow if we grow the chain.
8401  *
8402  * TODO: Free the actual blocks in the file when we shrink the chain.
8403  *       Currently, the blocks are still in the file. So the file size
8404  *       doesn't shrink even if we shrink streams.
8405  */
8406 BOOL SmallBlockChainStream_SetSize(
8407                 SmallBlockChainStream* This,
8408                 ULARGE_INTEGER    newSize)
8409 {
8410   ULARGE_INTEGER size = SmallBlockChainStream_GetSize(This);
8411 
8412   if (newSize.u.LowPart == size.u.LowPart)
8413     return TRUE;
8414 
8415   if (newSize.u.LowPart < size.u.LowPart)
8416   {
8417     SmallBlockChainStream_Shrink(This, newSize);
8418   }
8419   else
8420   {
8421     SmallBlockChainStream_Enlarge(This, newSize);
8422   }
8423 
8424   return TRUE;
8425 }
8426 
8427 /******************************************************************************
8428  *       SmallBlockChainStream_GetCount
8429  *
8430  * Returns the number of small blocks that comprises this chain.
8431  * This is not the size of the stream as the last block may not be full!
8432  *
8433  */
8434 static ULONG SmallBlockChainStream_GetCount(SmallBlockChainStream* This)
8435 {
8436     ULONG blockIndex;
8437     ULONG count = 0;
8438 
8439     blockIndex = SmallBlockChainStream_GetHeadOfChain(This);
8440 
8441     while(blockIndex != BLOCK_END_OF_CHAIN)
8442     {
8443         count++;
8444 
8445         if(FAILED(SmallBlockChainStream_GetNextBlockInChain(This,
8446                         blockIndex, &blockIndex)))
8447             return 0;
8448     }
8449 
8450     return count;
8451 }
8452 
8453 /******************************************************************************
8454  *      SmallBlockChainStream_GetSize
8455  *
8456  * Returns the size of this chain.
8457  */
8458 static ULARGE_INTEGER SmallBlockChainStream_GetSize(SmallBlockChainStream* This)
8459 {
8460   DirEntry chainEntry;
8461 
8462   if(This->headOfStreamPlaceHolder != NULL)
8463   {
8464     ULARGE_INTEGER result;
8465     result.u.HighPart = 0;
8466 
8467     result.u.LowPart = SmallBlockChainStream_GetCount(This) *
8468         This->parentStorage->smallBlockSize;
8469 
8470     return result;
8471   }
8472 
8473   StorageImpl_ReadDirEntry(
8474     This->parentStorage,
8475     This->ownerDirEntry,
8476     &chainEntry);
8477 
8478   return chainEntry.size;
8479 }
8480 
8481 
8482 /************************************************************************
8483  * Miscellaneous storage functions
8484  ***********************************************************************/
8485 
8486 static HRESULT create_storagefile(
8487   LPCOLESTR pwcsName,
8488   DWORD       grfMode,
8489   DWORD       grfAttrs,
8490   STGOPTIONS* pStgOptions,
8491   REFIID      riid,
8492   void**      ppstgOpen)
8493 {
8494   StorageBaseImpl* newStorage = 0;
8495   HANDLE       hFile      = INVALID_HANDLE_VALUE;
8496   HRESULT        hr         = STG_E_INVALIDFLAG;
8497   DWORD          shareMode;
8498   DWORD          accessMode;
8499   DWORD          creationMode;
8500   DWORD          fileAttributes;
8501   WCHAR          tempFileName[MAX_PATH];
8502 
8503   if (ppstgOpen == 0)
8504     return STG_E_INVALIDPOINTER;
8505 
8506   if (pStgOptions->ulSectorSize != MIN_BIG_BLOCK_SIZE && pStgOptions->ulSectorSize != MAX_BIG_BLOCK_SIZE)
8507     return STG_E_INVALIDPARAMETER;
8508 
8509   /* if no share mode given then DENY_NONE is the default */
8510   if (STGM_SHARE_MODE(grfMode) == 0)
8511       grfMode |= STGM_SHARE_DENY_NONE;
8512 
8513   if ( FAILED( validateSTGM(grfMode) ))
8514     goto end;
8515 
8516   /* StgCreateDocFile seems to refuse readonly access, despite MSDN */
8517   switch(STGM_ACCESS_MODE(grfMode))
8518   {
8519   case STGM_WRITE:
8520   case STGM_READWRITE:
8521     break;
8522   default:
8523     goto end;
8524   }
8525 
8526   /* in direct mode, can only use SHARE_EXCLUSIVE */
8527   if (!(grfMode & STGM_TRANSACTED) && (STGM_SHARE_MODE(grfMode) != STGM_SHARE_EXCLUSIVE))
8528     goto end;
8529 
8530   /* but in transacted mode, any share mode is valid */
8531 
8532   /*
8533    * Generate a unique name.
8534    */
8535   if (pwcsName == 0)
8536   {
8537     WCHAR tempPath[MAX_PATH];
8538     static const WCHAR prefix[] = { 'S', 'T', 'O', 0 };
8539 
8540     memset(tempPath, 0, sizeof(tempPath));
8541     memset(tempFileName, 0, sizeof(tempFileName));
8542 
8543     if ((GetTempPathW(MAX_PATH, tempPath)) == 0 )
8544       tempPath[0] = '.';
8545 
8546     if (GetTempFileNameW(tempPath, prefix, 0, tempFileName) != 0)
8547       pwcsName = tempFileName;
8548     else
8549     {
8550       hr = STG_E_INSUFFICIENTMEMORY;
8551       goto end;
8552     }
8553 
8554     creationMode = TRUNCATE_EXISTING;
8555   }
8556   else
8557   {
8558     creationMode = GetCreationModeFromSTGM(grfMode);
8559   }
8560 
8561   /*
8562    * Interpret the STGM value grfMode
8563    */
8564   shareMode    = GetShareModeFromSTGM(grfMode);
8565   accessMode   = GetAccessModeFromSTGM(grfMode);
8566 
8567   if (grfMode & STGM_DELETEONRELEASE)
8568     fileAttributes = FILE_FLAG_RANDOM_ACCESS | FILE_FLAG_DELETE_ON_CLOSE;
8569   else
8570     fileAttributes = FILE_ATTRIBUTE_NORMAL | FILE_FLAG_RANDOM_ACCESS;
8571 
8572   *ppstgOpen = 0;
8573 
8574   hFile = CreateFileW(pwcsName,
8575                         accessMode,
8576                         shareMode,
8577                         NULL,
8578                         creationMode,
8579                         fileAttributes,
8580                         0);
8581 
8582   if (hFile == INVALID_HANDLE_VALUE)
8583   {
8584     if(GetLastError() == ERROR_FILE_EXISTS)
8585       hr = STG_E_FILEALREADYEXISTS;
8586     else
8587       hr = E_FAIL;
8588     goto end;
8589   }
8590 
8591   /*
8592    * Allocate and initialize the new IStorage object.
8593    */
8594   hr = Storage_Construct(
8595          hFile,
8596         pwcsName,
8597          NULL,
8598          grfMode,
8599          TRUE,
8600          TRUE,
8601          pStgOptions->ulSectorSize,
8602          &newStorage);
8603 
8604   if (FAILED(hr))
8605   {
8606     goto end;
8607   }
8608 
8609   hr = IStorage_QueryInterface(&newStorage->IStorage_iface, riid, ppstgOpen);
8610   IStorage_Release(&newStorage->IStorage_iface);
8611 
8612 end:
8613   TRACE("<-- %p  r = %08x\n", *ppstgOpen, hr);
8614 
8615   return hr;
8616 }
8617 
8618 /******************************************************************************
8619  *    StgCreateDocfile  [OLE32.@]
8620  * Creates a new compound file storage object
8621  *
8622  * PARAMS
8623  *  pwcsName  [ I] Unicode string with filename (can be relative or NULL)
8624  *  grfMode   [ I] Access mode for opening the new storage object (see STGM_ constants)
8625  *  reserved  [ ?] unused?, usually 0
8626  *  ppstgOpen [IO] A pointer to IStorage pointer to the new onject
8627  *
8628  * RETURNS
8629  *  S_OK if the file was successfully created
8630  *  some STG_E_ value if error
8631  * NOTES
8632  *  if pwcsName is NULL, create file with new unique name
8633  *  the function can returns
8634  *  STG_S_CONVERTED if the specified file was successfully converted to storage format
8635  *  (unrealized now)
8636  */
8637 HRESULT WINAPI StgCreateDocfile(
8638   LPCOLESTR pwcsName,
8639   DWORD       grfMode,
8640   DWORD       reserved,
8641   IStorage  **ppstgOpen)
8642 {
8643   STGOPTIONS stgoptions = {1, 0, 512};
8644 
8645   TRACE("(%s, %x, %d, %p)\n",
8646 	debugstr_w(pwcsName), grfMode,
8647 	reserved, ppstgOpen);
8648 
8649   if (ppstgOpen == 0)
8650     return STG_E_INVALIDPOINTER;
8651   if (reserved != 0)
8652     return STG_E_INVALIDPARAMETER;
8653 
8654   return create_storagefile(pwcsName, grfMode, 0, &stgoptions, &IID_IStorage, (void**)ppstgOpen);
8655 }
8656 
8657 /******************************************************************************
8658  *              StgCreateStorageEx        [OLE32.@]
8659  */
8660 HRESULT WINAPI StgCreateStorageEx(const WCHAR* pwcsName, DWORD grfMode, DWORD stgfmt, DWORD grfAttrs, STGOPTIONS* pStgOptions, void* reserved, REFIID riid, void** ppObjectOpen)
8661 {
8662     TRACE("(%s, %x, %x, %x, %p, %p, %p, %p)\n", debugstr_w(pwcsName),
8663           grfMode, stgfmt, grfAttrs, pStgOptions, reserved, riid, ppObjectOpen);
8664 
8665     if (stgfmt != STGFMT_FILE && grfAttrs != 0)
8666     {
8667         ERR("grfAttrs must be 0 if stgfmt != STGFMT_FILE\n");
8668         return STG_E_INVALIDPARAMETER;
8669     }
8670 
8671     if (stgfmt == STGFMT_FILE && grfAttrs != 0 && grfAttrs != FILE_FLAG_NO_BUFFERING)
8672     {
8673         ERR("grfAttrs must be 0 or FILE_FLAG_NO_BUFFERING if stgfmt == STGFMT_FILE\n");
8674         return STG_E_INVALIDPARAMETER;
8675     }
8676 
8677     if (stgfmt == STGFMT_FILE)
8678     {
8679         ERR("Cannot use STGFMT_FILE - this is NTFS only\n");
8680         return STG_E_INVALIDPARAMETER;
8681     }
8682 
8683     if (stgfmt == STGFMT_STORAGE || stgfmt == STGFMT_DOCFILE)
8684     {
8685         STGOPTIONS defaultOptions = {1, 0, 512};
8686 
8687         if (!pStgOptions) pStgOptions = &defaultOptions;
8688         return create_storagefile(pwcsName, grfMode, grfAttrs, pStgOptions, riid, ppObjectOpen);
8689     }
8690 
8691 
8692     ERR("Invalid stgfmt argument\n");
8693     return STG_E_INVALIDPARAMETER;
8694 }
8695 
8696 /******************************************************************************
8697  *              StgCreatePropSetStg       [OLE32.@]
8698  */
8699 HRESULT WINAPI StgCreatePropSetStg(IStorage *pstg, DWORD reserved,
8700  IPropertySetStorage **propset)
8701 {
8702     TRACE("(%p, 0x%x, %p)\n", pstg, reserved, propset);
8703     if (reserved)
8704         return STG_E_INVALIDPARAMETER;
8705 
8706     return IStorage_QueryInterface(pstg, &IID_IPropertySetStorage, (void**)propset);
8707 }
8708 
8709 /******************************************************************************
8710  *              StgOpenStorageEx      [OLE32.@]
8711  */
8712 HRESULT WINAPI StgOpenStorageEx(const WCHAR* pwcsName, DWORD grfMode, DWORD stgfmt, DWORD grfAttrs, STGOPTIONS* pStgOptions, void* reserved, REFIID riid, void** ppObjectOpen)
8713 {
8714     TRACE("(%s, %x, %x, %x, %p, %p, %p, %p)\n", debugstr_w(pwcsName),
8715           grfMode, stgfmt, grfAttrs, pStgOptions, reserved, riid, ppObjectOpen);
8716 
8717     if (stgfmt != STGFMT_DOCFILE && grfAttrs != 0)
8718     {
8719         ERR("grfAttrs must be 0 if stgfmt != STGFMT_DOCFILE\n");
8720         return STG_E_INVALIDPARAMETER;
8721     }
8722 
8723     switch (stgfmt)
8724     {
8725     case STGFMT_FILE:
8726         ERR("Cannot use STGFMT_FILE - this is NTFS only\n");
8727         return STG_E_INVALIDPARAMETER;
8728 
8729     case STGFMT_STORAGE:
8730         break;
8731 
8732     case STGFMT_DOCFILE:
8733         if (grfAttrs && grfAttrs != FILE_FLAG_NO_BUFFERING)
8734         {
8735             ERR("grfAttrs must be 0 or FILE_FLAG_NO_BUFFERING if stgfmt == STGFMT_DOCFILE\n");
8736             return STG_E_INVALIDPARAMETER;
8737         }
8738         FIXME("Stub: calling StgOpenStorage, but ignoring pStgOptions and grfAttrs\n");
8739         break;
8740 
8741     case STGFMT_ANY:
8742         WARN("STGFMT_ANY assuming storage\n");
8743         break;
8744 
8745     default:
8746         return STG_E_INVALIDPARAMETER;
8747     }
8748 
8749     return StgOpenStorage(pwcsName, NULL, grfMode, NULL, 0, (IStorage **)ppObjectOpen);
8750 }
8751 
8752 
8753 /******************************************************************************
8754  *              StgOpenStorage        [OLE32.@]
8755  */
8756 HRESULT WINAPI StgOpenStorage(
8757   const OLECHAR *pwcsName,
8758   IStorage      *pstgPriority,
8759   DWORD          grfMode,
8760   SNB            snbExclude,
8761   DWORD          reserved,
8762   IStorage     **ppstgOpen)
8763 {
8764   StorageBaseImpl* newStorage = 0;
8765   HRESULT        hr = S_OK;
8766   HANDLE         hFile = 0;
8767   DWORD          shareMode;
8768   DWORD          accessMode;
8769   LPWSTR         temp_name = NULL;
8770 
8771   TRACE("(%s, %p, %x, %p, %d, %p)\n",
8772 	debugstr_w(pwcsName), pstgPriority, grfMode,
8773 	snbExclude, reserved, ppstgOpen);
8774 
8775   if (pstgPriority)
8776   {
8777     /* FIXME: Copy ILockBytes instead? But currently for STGM_PRIORITY it'll be read-only. */
8778     hr = StorageBaseImpl_GetFilename((StorageBaseImpl*)pstgPriority, &temp_name);
8779     if (FAILED(hr)) goto end;
8780     pwcsName = temp_name;
8781     TRACE("using filename %s\n", debugstr_w(temp_name));
8782   }
8783 
8784   if (pwcsName == 0)
8785   {
8786     hr = STG_E_INVALIDNAME;
8787     goto end;
8788   }
8789 
8790   if (ppstgOpen == 0)
8791   {
8792     hr = STG_E_INVALIDPOINTER;
8793     goto end;
8794   }
8795 
8796   if (reserved)
8797   {
8798     hr = STG_E_INVALIDPARAMETER;
8799     goto end;
8800   }
8801 
8802   if (grfMode & STGM_PRIORITY)
8803   {
8804     if (grfMode & (STGM_TRANSACTED|STGM_SIMPLE|STGM_NOSCRATCH|STGM_NOSNAPSHOT))
8805       return STG_E_INVALIDFLAG;
8806     if (grfMode & STGM_DELETEONRELEASE)
8807       return STG_E_INVALIDFUNCTION;
8808     if(STGM_ACCESS_MODE(grfMode) != STGM_READ)
8809       return STG_E_INVALIDFLAG;
8810     grfMode &= ~0xf0; /* remove the existing sharing mode */
8811     grfMode |= STGM_SHARE_DENY_NONE;
8812   }
8813 
8814   /*
8815    * Validate the sharing mode
8816    */
8817   if (grfMode & STGM_DIRECT_SWMR)
8818   {
8819     if ((STGM_SHARE_MODE(grfMode) != STGM_SHARE_DENY_WRITE) &&
8820         (STGM_SHARE_MODE(grfMode) != STGM_SHARE_DENY_NONE))
8821     {
8822       hr = STG_E_INVALIDFLAG;
8823       goto end;
8824     }
8825   }
8826   else if (!(grfMode & (STGM_TRANSACTED|STGM_PRIORITY)))
8827     switch(STGM_SHARE_MODE(grfMode))
8828     {
8829       case STGM_SHARE_EXCLUSIVE:
8830       case STGM_SHARE_DENY_WRITE:
8831         break;
8832       default:
8833         hr = STG_E_INVALIDFLAG;
8834         goto end;
8835     }
8836 
8837   if ( FAILED( validateSTGM(grfMode) ) ||
8838        (grfMode&STGM_CREATE))
8839   {
8840     hr = STG_E_INVALIDFLAG;
8841     goto end;
8842   }
8843 
8844   /* shared reading requires transacted or single writer mode */
8845   if( STGM_SHARE_MODE(grfMode) == STGM_SHARE_DENY_WRITE &&
8846       STGM_ACCESS_MODE(grfMode) == STGM_READWRITE &&
8847      !(grfMode & STGM_TRANSACTED) && !(grfMode & STGM_DIRECT_SWMR))
8848   {
8849     hr = STG_E_INVALIDFLAG;
8850     goto end;
8851   }
8852 
8853   /*
8854    * Interpret the STGM value grfMode
8855    */
8856   shareMode    = GetShareModeFromSTGM(grfMode);
8857   accessMode   = GetAccessModeFromSTGM(grfMode);
8858 
8859   *ppstgOpen = 0;
8860 
8861   hFile = CreateFileW( pwcsName,
8862                        accessMode,
8863                        shareMode,
8864                        NULL,
8865                        OPEN_EXISTING,
8866                        FILE_ATTRIBUTE_NORMAL | FILE_FLAG_RANDOM_ACCESS,
8867                        0);
8868 
8869   if (hFile==INVALID_HANDLE_VALUE)
8870   {
8871     DWORD last_error = GetLastError();
8872 
8873     hr = E_FAIL;
8874 
8875     switch (last_error)
8876     {
8877       case ERROR_FILE_NOT_FOUND:
8878         hr = STG_E_FILENOTFOUND;
8879         break;
8880 
8881       case ERROR_PATH_NOT_FOUND:
8882         hr = STG_E_PATHNOTFOUND;
8883         break;
8884 
8885       case ERROR_ACCESS_DENIED:
8886       case ERROR_WRITE_PROTECT:
8887         hr = STG_E_ACCESSDENIED;
8888         break;
8889 
8890       case ERROR_SHARING_VIOLATION:
8891         hr = STG_E_SHAREVIOLATION;
8892         break;
8893 
8894       default:
8895         hr = E_FAIL;
8896     }
8897 
8898     goto end;
8899   }
8900 
8901   /*
8902    * Refuse to open the file if it's too small to be a structured storage file
8903    * FIXME: verify the file when reading instead of here
8904    */
8905   if (GetFileSize(hFile, NULL) < HEADER_SIZE)
8906   {
8907     CloseHandle(hFile);
8908     hr = STG_E_FILEALREADYEXISTS;
8909     goto end;
8910   }
8911 
8912   /*
8913    * Allocate and initialize the new IStorage object.
8914    */
8915   hr = Storage_Construct(
8916          hFile,
8917          pwcsName,
8918          NULL,
8919          grfMode,
8920          TRUE,
8921          FALSE,
8922          512,
8923          &newStorage);
8924 
8925   if (FAILED(hr))
8926   {
8927     /*
8928      * According to the docs if the file is not a storage, return STG_E_FILEALREADYEXISTS
8929      */
8930     if(hr == STG_E_INVALIDHEADER)
8931 	hr = STG_E_FILEALREADYEXISTS;
8932     goto end;
8933   }
8934 
8935   *ppstgOpen = &newStorage->IStorage_iface;
8936 
8937 end:
8938   CoTaskMemFree(temp_name);
8939   if (pstgPriority) IStorage_Release(pstgPriority);
8940   TRACE("<-- %08x, IStorage %p\n", hr, ppstgOpen ? *ppstgOpen : NULL);
8941   return hr;
8942 }
8943 
8944 /******************************************************************************
8945  *    StgCreateDocfileOnILockBytes    [OLE32.@]
8946  */
8947 HRESULT WINAPI StgCreateDocfileOnILockBytes(
8948       ILockBytes *plkbyt,
8949       DWORD grfMode,
8950       DWORD reserved,
8951       IStorage** ppstgOpen)
8952 {
8953   StorageBaseImpl* newStorage = 0;
8954   HRESULT        hr         = S_OK;
8955 
8956   if ((ppstgOpen == 0) || (plkbyt == 0))
8957     return STG_E_INVALIDPOINTER;
8958 
8959   /*
8960    * Allocate and initialize the new IStorage object.
8961    */
8962   hr = Storage_Construct(
8963          0,
8964         0,
8965          plkbyt,
8966          grfMode,
8967          FALSE,
8968          TRUE,
8969          512,
8970          &newStorage);
8971 
8972   if (FAILED(hr))
8973   {
8974     return hr;
8975   }
8976 
8977   *ppstgOpen = &newStorage->IStorage_iface;
8978 
8979   return hr;
8980 }
8981 
8982 /******************************************************************************
8983  *    StgOpenStorageOnILockBytes    [OLE32.@]
8984  */
8985 HRESULT WINAPI StgOpenStorageOnILockBytes(
8986       ILockBytes *plkbyt,
8987       IStorage *pstgPriority,
8988       DWORD grfMode,
8989       SNB snbExclude,
8990       DWORD reserved,
8991       IStorage **ppstgOpen)
8992 {
8993   StorageBaseImpl* newStorage = 0;
8994   HRESULT        hr = S_OK;
8995 
8996   if ((plkbyt == 0) || (ppstgOpen == 0))
8997     return STG_E_INVALIDPOINTER;
8998 
8999   if ( FAILED( validateSTGM(grfMode) ))
9000     return STG_E_INVALIDFLAG;
9001 
9002   *ppstgOpen = 0;
9003 
9004   /*
9005    * Allocate and initialize the new IStorage object.
9006    */
9007   hr = Storage_Construct(
9008          0,
9009          0,
9010          plkbyt,
9011          grfMode,
9012          FALSE,
9013          FALSE,
9014          512,
9015          &newStorage);
9016 
9017   if (FAILED(hr))
9018   {
9019     return hr;
9020   }
9021 
9022   *ppstgOpen = &newStorage->IStorage_iface;
9023 
9024   return hr;
9025 }
9026 
9027 /******************************************************************************
9028  *              StgSetTimes [ole32.@]
9029  *              StgSetTimes [OLE32.@]
9030  *
9031  *
9032  */
9033 HRESULT WINAPI StgSetTimes(OLECHAR const *str, FILETIME const *pctime,
9034                            FILETIME const *patime, FILETIME const *pmtime)
9035 {
9036   IStorage *stg = NULL;
9037   HRESULT r;
9038 
9039   TRACE("%s %p %p %p\n", debugstr_w(str), pctime, patime, pmtime);
9040 
9041   r = StgOpenStorage(str, NULL, STGM_READWRITE | STGM_SHARE_DENY_WRITE,
9042                      0, 0, &stg);
9043   if( SUCCEEDED(r) )
9044   {
9045     r = IStorage_SetElementTimes(stg, NULL, pctime, patime, pmtime);
9046     IStorage_Release(stg);
9047   }
9048 
9049   return r;
9050 }
9051 
9052 /******************************************************************************
9053  *              StgIsStorageILockBytes        [OLE32.@]
9054  *
9055  * Determines if the ILockBytes contains a storage object.
9056  */
9057 HRESULT WINAPI StgIsStorageILockBytes(ILockBytes *plkbyt)
9058 {
9059   BYTE sig[sizeof(STORAGE_magic)];
9060   ULARGE_INTEGER offset;
9061   ULONG read = 0;
9062 
9063   offset.u.HighPart = 0;
9064   offset.u.LowPart  = 0;
9065 
9066   ILockBytes_ReadAt(plkbyt, offset, sig, sizeof(sig), &read);
9067 
9068   if (read == sizeof(sig) && memcmp(sig, STORAGE_magic, sizeof(sig)) == 0)
9069     return S_OK;
9070 
9071   return S_FALSE;
9072 }
9073 
9074 /******************************************************************************
9075  *              WriteClassStg        [OLE32.@]
9076  *
9077  * This method will store the specified CLSID in the specified storage object
9078  */
9079 HRESULT WINAPI WriteClassStg(IStorage* pStg, REFCLSID rclsid)
9080 {
9081   if(!pStg)
9082     return E_INVALIDARG;
9083 
9084   if(!rclsid)
9085     return STG_E_INVALIDPOINTER;
9086 
9087   return IStorage_SetClass(pStg, rclsid);
9088 }
9089 
9090 /***********************************************************************
9091  *    ReadClassStg (OLE32.@)
9092  *
9093  * This method reads the CLSID previously written to a storage object with
9094  * the WriteClassStg.
9095  *
9096  * PARAMS
9097  *  pstg    [I] IStorage pointer
9098  *  pclsid  [O] Pointer to where the CLSID is written
9099  *
9100  * RETURNS
9101  *  Success: S_OK.
9102  *  Failure: HRESULT code.
9103  */
9104 HRESULT WINAPI ReadClassStg(IStorage *pstg,CLSID *pclsid){
9105 
9106     STATSTG pstatstg;
9107     HRESULT hRes;
9108 
9109     TRACE("(%p, %p)\n", pstg, pclsid);
9110 
9111     if(!pstg || !pclsid)
9112         return E_INVALIDARG;
9113 
9114    /*
9115     * read a STATSTG structure (contains the clsid) from the storage
9116     */
9117     hRes=IStorage_Stat(pstg,&pstatstg,STATFLAG_NONAME);
9118 
9119     if(SUCCEEDED(hRes))
9120         *pclsid=pstatstg.clsid;
9121 
9122     return hRes;
9123 }
9124 
9125 /***********************************************************************
9126  *    OleLoadFromStream (OLE32.@)
9127  *
9128  * This function loads an object from stream
9129  */
9130 HRESULT  WINAPI OleLoadFromStream(IStream *pStm,REFIID iidInterface,void** ppvObj)
9131 {
9132     CLSID	clsid;
9133     HRESULT	res;
9134     LPPERSISTSTREAM	xstm;
9135 
9136     TRACE("(%p,%s,%p)\n",pStm,debugstr_guid(iidInterface),ppvObj);
9137 
9138     res=ReadClassStm(pStm,&clsid);
9139     if (FAILED(res))
9140 	return res;
9141     res=CoCreateInstance(&clsid,NULL,CLSCTX_INPROC_SERVER,iidInterface,ppvObj);
9142     if (FAILED(res))
9143 	return res;
9144     res=IUnknown_QueryInterface((IUnknown*)*ppvObj,&IID_IPersistStream,(LPVOID*)&xstm);
9145     if (FAILED(res)) {
9146 	IUnknown_Release((IUnknown*)*ppvObj);
9147 	return res;
9148     }
9149     res=IPersistStream_Load(xstm,pStm);
9150     IPersistStream_Release(xstm);
9151     /* FIXME: all refcounts ok at this point? I think they should be:
9152      * 		pStm	: unchanged
9153      *		ppvObj	: 1
9154      *		xstm	: 0 (released)
9155      */
9156     return res;
9157 }
9158 
9159 /***********************************************************************
9160  *    OleSaveToStream (OLE32.@)
9161  *
9162  * This function saves an object with the IPersistStream interface on it
9163  * to the specified stream.
9164  */
9165 HRESULT  WINAPI OleSaveToStream(IPersistStream *pPStm,IStream *pStm)
9166 {
9167 
9168     CLSID clsid;
9169     HRESULT res;
9170 
9171     TRACE("(%p,%p)\n",pPStm,pStm);
9172 
9173     res=IPersistStream_GetClassID(pPStm,&clsid);
9174 
9175     if (SUCCEEDED(res)){
9176 
9177         res=WriteClassStm(pStm,&clsid);
9178 
9179         if (SUCCEEDED(res))
9180 
9181             res=IPersistStream_Save(pPStm,pStm,TRUE);
9182     }
9183 
9184     TRACE("Finished Save\n");
9185     return res;
9186 }
9187 
9188 /*************************************************************************
9189  * STORAGE_CreateOleStream [Internal]
9190  *
9191  * Creates the "\001OLE" stream in the IStorage if necessary.
9192  *
9193  * PARAMS
9194  *     storage     [I] Dest storage to create the stream in
9195  *     flags       [I] flags to be set for newly created stream
9196  *
9197  * RETURNS
9198  *     HRESULT return value
9199  *
9200  * NOTES
9201  *
9202  *     This stream is still unknown, MS Word seems to have extra data
9203  *     but since the data is stored in the OLESTREAM there should be
9204  *     no need to recreate the stream.  If the stream is manually
9205  *     deleted it will create it with this default data.
9206  *
9207  */
9208 HRESULT STORAGE_CreateOleStream(IStorage *storage, DWORD flags)
9209 {
9210     static const WCHAR stream_1oleW[] = {1,'O','l','e',0};
9211     static const DWORD version_magic = 0x02000001;
9212     IStream *stream;
9213     HRESULT hr;
9214 
9215     hr = IStorage_CreateStream(storage, stream_1oleW, STGM_WRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &stream);
9216     if (hr == S_OK)
9217     {
9218         struct empty_1ole_stream {
9219             DWORD version_magic;
9220             DWORD flags;
9221             DWORD update_options;
9222             DWORD reserved;
9223             DWORD mon_stream_size;
9224         };
9225         struct empty_1ole_stream stream_data;
9226 
9227         stream_data.version_magic = version_magic;
9228         stream_data.flags = flags;
9229         stream_data.update_options = 0;
9230         stream_data.reserved = 0;
9231         stream_data.mon_stream_size = 0;
9232 
9233         hr = IStream_Write(stream, &stream_data, sizeof(stream_data), NULL);
9234         IStream_Release(stream);
9235     }
9236 
9237     return hr;
9238 }
9239 
9240 /* write a string to a stream, preceded by its length */
9241 static HRESULT STREAM_WriteString( IStream *stm, LPCWSTR string )
9242 {
9243     HRESULT r;
9244     LPSTR str;
9245     DWORD len = 0;
9246 
9247     if( string )
9248         len = WideCharToMultiByte( CP_ACP, 0, string, -1, NULL, 0, NULL, NULL);
9249     r = IStream_Write( stm, &len, sizeof(len), NULL);
9250     if( FAILED( r ) )
9251         return r;
9252     if(len == 0)
9253         return r;
9254     str = CoTaskMemAlloc( len );
9255     WideCharToMultiByte( CP_ACP, 0, string, -1, str, len, NULL, NULL);
9256     r = IStream_Write( stm, str, len, NULL);
9257     CoTaskMemFree( str );
9258     return r;
9259 }
9260 
9261 /* read a string preceded by its length from a stream */
9262 static HRESULT STREAM_ReadString( IStream *stm, LPWSTR *string )
9263 {
9264     HRESULT r;
9265     DWORD len, count = 0;
9266     LPSTR str;
9267     LPWSTR wstr;
9268 
9269     r = IStream_Read( stm, &len, sizeof(len), &count );
9270     if( FAILED( r ) )
9271         return r;
9272     if( count != sizeof(len) )
9273         return E_OUTOFMEMORY;
9274 
9275     TRACE("%d bytes\n",len);
9276 
9277     str = CoTaskMemAlloc( len );
9278     if( !str )
9279         return E_OUTOFMEMORY;
9280     count = 0;
9281     r = IStream_Read( stm, str, len, &count );
9282     if( FAILED( r ) )
9283     {
9284         CoTaskMemFree( str );
9285         return r;
9286     }
9287     if( count != len )
9288     {
9289         CoTaskMemFree( str );
9290         return E_OUTOFMEMORY;
9291     }
9292 
9293     TRACE("Read string %s\n",debugstr_an(str,len));
9294 
9295     len = MultiByteToWideChar( CP_ACP, 0, str, count, NULL, 0 );
9296     wstr = CoTaskMemAlloc( (len + 1)*sizeof (WCHAR) );
9297     if( wstr )
9298     {
9299          MultiByteToWideChar( CP_ACP, 0, str, count, wstr, len );
9300          wstr[len] = 0;
9301     }
9302     CoTaskMemFree( str );
9303 
9304     *string = wstr;
9305 
9306     return r;
9307 }
9308 
9309 
9310 static HRESULT STORAGE_WriteCompObj( LPSTORAGE pstg, CLSID *clsid,
9311     LPCWSTR lpszUserType, LPCWSTR szClipName, LPCWSTR szProgIDName )
9312 {
9313     IStream *pstm;
9314     HRESULT r = S_OK;
9315     static const WCHAR szwStreamName[] = {1, 'C', 'o', 'm', 'p', 'O', 'b', 'j', 0};
9316 
9317     static const BYTE unknown1[12] =
9318        { 0x01, 0x00, 0xFE, 0xFF, 0x03, 0x0A, 0x00, 0x00,
9319          0xFF, 0xFF, 0xFF, 0xFF};
9320     static const BYTE unknown2[16] =
9321        { 0xF4, 0x39, 0xB2, 0x71, 0x00, 0x00, 0x00, 0x00,
9322          0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
9323 
9324     TRACE("%p %s %s %s %s\n", pstg, debugstr_guid(clsid),
9325            debugstr_w(lpszUserType), debugstr_w(szClipName),
9326            debugstr_w(szProgIDName));
9327 
9328     /*  Create a CompObj stream */
9329     r = IStorage_CreateStream(pstg, szwStreamName,
9330         STGM_CREATE | STGM_WRITE  | STGM_SHARE_EXCLUSIVE, 0, 0, &pstm );
9331     if( FAILED (r) )
9332         return r;
9333 
9334     /* Write CompObj Structure to stream */
9335     r = IStream_Write(pstm, unknown1, sizeof(unknown1), NULL);
9336 
9337     if( SUCCEEDED( r ) )
9338         r = WriteClassStm( pstm, clsid );
9339 
9340     if( SUCCEEDED( r ) )
9341         r = STREAM_WriteString( pstm, lpszUserType );
9342     if( SUCCEEDED( r ) )
9343         r = STREAM_WriteString( pstm, szClipName );
9344     if( SUCCEEDED( r ) )
9345         r = STREAM_WriteString( pstm, szProgIDName );
9346     if( SUCCEEDED( r ) )
9347         r = IStream_Write(pstm, unknown2, sizeof(unknown2), NULL);
9348 
9349     IStream_Release( pstm );
9350 
9351     return r;
9352 }
9353 
9354 /***********************************************************************
9355  *               WriteFmtUserTypeStg (OLE32.@)
9356  */
9357 HRESULT WINAPI WriteFmtUserTypeStg(
9358 	  LPSTORAGE pstg, CLIPFORMAT cf, LPOLESTR lpszUserType)
9359 {
9360     STATSTG stat;
9361     HRESULT r;
9362     WCHAR szwClipName[0x40];
9363     CLSID clsid;
9364     LPWSTR wstrProgID = NULL;
9365     DWORD n;
9366 
9367     TRACE("(%p,%x,%s)\n",pstg,cf,debugstr_w(lpszUserType));
9368 
9369     /* get the clipboard format name */
9370     if( cf )
9371     {
9372         n = GetClipboardFormatNameW(cf, szwClipName, ARRAY_SIZE(szwClipName));
9373         szwClipName[n]=0;
9374     }
9375 
9376     TRACE("Clipboard name is %s\n", debugstr_w(szwClipName));
9377 
9378     r = IStorage_Stat(pstg, &stat, STATFLAG_NONAME);
9379     if(SUCCEEDED(r))
9380         clsid = stat.clsid;
9381     else
9382         clsid = CLSID_NULL;
9383 
9384     ProgIDFromCLSID(&clsid, &wstrProgID);
9385 
9386     TRACE("progid is %s\n",debugstr_w(wstrProgID));
9387 
9388     r = STORAGE_WriteCompObj( pstg, &clsid, lpszUserType,
9389             cf ? szwClipName : NULL, wstrProgID );
9390 
9391     CoTaskMemFree(wstrProgID);
9392 
9393     return r;
9394 }
9395 
9396 
9397 /******************************************************************************
9398  *              ReadFmtUserTypeStg        [OLE32.@]
9399  */
9400 HRESULT WINAPI ReadFmtUserTypeStg (LPSTORAGE pstg, CLIPFORMAT* pcf, LPOLESTR* lplpszUserType)
9401 {
9402     HRESULT r;
9403     IStream *stm = 0;
9404     static const WCHAR szCompObj[] = { 1, 'C','o','m','p','O','b','j', 0 };
9405     unsigned char unknown1[12];
9406     unsigned char unknown2[16];
9407     DWORD count;
9408     LPWSTR szProgIDName = NULL, szCLSIDName = NULL, szOleTypeName = NULL;
9409     CLSID clsid;
9410 
9411     TRACE("(%p,%p,%p)\n", pstg, pcf, lplpszUserType);
9412 
9413     r = IStorage_OpenStream( pstg, szCompObj, NULL,
9414                     STGM_READ | STGM_SHARE_EXCLUSIVE, 0, &stm );
9415     if( FAILED ( r ) )
9416     {
9417         WARN("Failed to open stream r = %08x\n", r);
9418         return r;
9419     }
9420 
9421     /* read the various parts of the structure */
9422     r = IStream_Read( stm, unknown1, sizeof(unknown1), &count );
9423     if( FAILED( r ) || ( count != sizeof(unknown1) ) )
9424         goto end;
9425     r = ReadClassStm( stm, &clsid );
9426     if( FAILED( r ) )
9427         goto end;
9428 
9429     r = STREAM_ReadString( stm, &szCLSIDName );
9430     if( FAILED( r ) )
9431         goto end;
9432 
9433     r = STREAM_ReadString( stm, &szOleTypeName );
9434     if( FAILED( r ) )
9435         goto end;
9436 
9437     r = STREAM_ReadString( stm, &szProgIDName );
9438     if( FAILED( r ) )
9439         goto end;
9440 
9441     r = IStream_Read( stm, unknown2, sizeof(unknown2), &count );
9442     if( FAILED( r ) || ( count != sizeof(unknown2) ) )
9443         goto end;
9444 
9445     /* ok, success... now we just need to store what we found */
9446     if( pcf )
9447         *pcf = RegisterClipboardFormatW( szOleTypeName );
9448 
9449     if( lplpszUserType )
9450     {
9451         *lplpszUserType = szCLSIDName;
9452         szCLSIDName = NULL;
9453     }
9454 
9455 end:
9456     CoTaskMemFree( szCLSIDName );
9457     CoTaskMemFree( szOleTypeName );
9458     CoTaskMemFree( szProgIDName );
9459     IStream_Release( stm );
9460 
9461     return r;
9462 }
9463 
9464 /******************************************************************************
9465  * StgIsStorageFile [OLE32.@]
9466  * Verify if the file contains a storage object
9467  *
9468  * PARAMS
9469  *  fn      [ I] Filename
9470  *
9471  * RETURNS
9472  *  S_OK    if file has magic bytes as a storage object
9473  *  S_FALSE if file is not storage
9474  */
9475 HRESULT WINAPI
9476 StgIsStorageFile(LPCOLESTR fn)
9477 {
9478 	HANDLE		hf;
9479 	BYTE		magic[8];
9480 	DWORD		bytes_read;
9481 
9482 	TRACE("%s\n", debugstr_w(fn));
9483 	hf = CreateFileW(fn, GENERIC_READ,
9484 	                 FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE,
9485 	                 NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
9486 
9487 	if (hf == INVALID_HANDLE_VALUE)
9488 		return STG_E_FILENOTFOUND;
9489 
9490 	if (!ReadFile(hf, magic, 8, &bytes_read, NULL))
9491 	{
9492 		WARN(" unable to read file\n");
9493 		CloseHandle(hf);
9494 		return S_FALSE;
9495 	}
9496 
9497 	CloseHandle(hf);
9498 
9499 	if (bytes_read != 8) {
9500 		TRACE(" too short\n");
9501 		return S_FALSE;
9502 	}
9503 
9504 	if (!memcmp(magic,STORAGE_magic,8)) {
9505 		TRACE(" -> YES\n");
9506 		return S_OK;
9507 	}
9508 
9509 	TRACE(" -> Invalid header.\n");
9510 	return S_FALSE;
9511 }
9512 
9513 /***********************************************************************
9514  *		WriteClassStm (OLE32.@)
9515  *
9516  * Writes a CLSID to a stream.
9517  *
9518  * PARAMS
9519  *  pStm   [I] Stream to write to.
9520  *  rclsid [I] CLSID to write.
9521  *
9522  * RETURNS
9523  *  Success: S_OK.
9524  *  Failure: HRESULT code.
9525  */
9526 HRESULT WINAPI WriteClassStm(IStream *pStm,REFCLSID rclsid)
9527 {
9528     TRACE("(%p,%p)\n",pStm,rclsid);
9529 
9530     if (!pStm || !rclsid)
9531         return E_INVALIDARG;
9532 
9533     return IStream_Write(pStm,rclsid,sizeof(CLSID),NULL);
9534 }
9535 
9536 /***********************************************************************
9537  *		ReadClassStm (OLE32.@)
9538  *
9539  * Reads a CLSID from a stream.
9540  *
9541  * PARAMS
9542  *  pStm   [I] Stream to read from.
9543  *  rclsid [O] CLSID to read.
9544  *
9545  * RETURNS
9546  *  Success: S_OK.
9547  *  Failure: HRESULT code.
9548  */
9549 HRESULT WINAPI ReadClassStm(IStream *pStm,CLSID *pclsid)
9550 {
9551     ULONG nbByte;
9552     HRESULT res;
9553 
9554     TRACE("(%p,%p)\n",pStm,pclsid);
9555 
9556     if (!pStm || !pclsid)
9557         return E_INVALIDARG;
9558 
9559     /* clear the output args */
9560     *pclsid = CLSID_NULL;
9561 
9562     res = IStream_Read(pStm, pclsid, sizeof(CLSID), &nbByte);
9563 
9564     if (FAILED(res))
9565         return res;
9566 
9567     if (nbByte != sizeof(CLSID))
9568         return STG_E_READFAULT;
9569     else
9570         return S_OK;
9571 }
9572 
9573 
9574 /************************************************************************
9575  * OleConvert Functions
9576  ***********************************************************************/
9577 
9578 #define OLESTREAM_ID 0x501
9579 #define OLESTREAM_MAX_STR_LEN 255
9580 
9581 /* OLESTREAM memory structure to use for Get and Put Routines */
9582 typedef struct
9583 {
9584     DWORD dwOleID;
9585     DWORD dwTypeID;
9586     DWORD dwOleTypeNameLength;
9587     CHAR  strOleTypeName[OLESTREAM_MAX_STR_LEN];
9588     CHAR  *pstrOleObjFileName;
9589     DWORD dwOleObjFileNameLength;
9590     DWORD dwMetaFileWidth;
9591     DWORD dwMetaFileHeight;
9592     CHAR  strUnknown[8]; /* don't know what this 8 byte information in OLE stream is. */
9593     DWORD dwDataLength;
9594     BYTE *pData;
9595 } OLECONVERT_OLESTREAM_DATA;
9596 
9597 /* CompObj Stream structure */
9598 typedef struct
9599 {
9600     BYTE byUnknown1[12];
9601     CLSID clsid;
9602     DWORD dwCLSIDNameLength;
9603     CHAR strCLSIDName[OLESTREAM_MAX_STR_LEN];
9604     DWORD dwOleTypeNameLength;
9605     CHAR strOleTypeName[OLESTREAM_MAX_STR_LEN];
9606     DWORD dwProgIDNameLength;
9607     CHAR strProgIDName[OLESTREAM_MAX_STR_LEN];
9608     BYTE byUnknown2[16];
9609 } OLECONVERT_ISTORAGE_COMPOBJ;
9610 
9611 /* Ole Presentation Stream structure */
9612 typedef struct
9613 {
9614     BYTE byUnknown1[28];
9615     DWORD dwExtentX;
9616     DWORD dwExtentY;
9617     DWORD dwSize;
9618     BYTE *pData;
9619 } OLECONVERT_ISTORAGE_OLEPRES;
9620 
9621 
9622 /*************************************************************************
9623  * OLECONVERT_LoadOLE10 [Internal]
9624  *
9625  * Loads the OLE10 STREAM to memory
9626  *
9627  * PARAMS
9628  *     pOleStream   [I] The OLESTREAM
9629  *     pData        [I] Data Structure for the OLESTREAM Data
9630  *
9631  * RETURNS
9632  *     Success:  S_OK
9633  *     Failure:  CONVERT10_E_OLESTREAM_GET for invalid Get
9634  *               CONVERT10_E_OLESTREAM_FMT if the OLEID is invalid
9635  *
9636  * NOTES
9637  *     This function is used by OleConvertOLESTREAMToIStorage only.
9638  *
9639  *     Memory allocated for pData must be freed by the caller
9640  */
9641 static HRESULT OLECONVERT_LoadOLE10(LPOLESTREAM pOleStream, OLECONVERT_OLESTREAM_DATA *pData, BOOL bStrem1)
9642 {
9643 	DWORD dwSize;
9644 	HRESULT hRes = S_OK;
9645 	int nTryCnt=0;
9646 	int max_try = 6;
9647 
9648 	pData->pData = NULL;
9649 	pData->pstrOleObjFileName = NULL;
9650 
9651 	for( nTryCnt=0;nTryCnt < max_try; nTryCnt++)
9652 	{
9653 	/* Get the OleID */
9654 	dwSize = pOleStream->lpstbl->Get(pOleStream, (void *)&(pData->dwOleID), sizeof(pData->dwOleID));
9655 	if(dwSize != sizeof(pData->dwOleID))
9656 	{
9657 		hRes = CONVERT10_E_OLESTREAM_GET;
9658 	}
9659 	else if(pData->dwOleID != OLESTREAM_ID)
9660 	{
9661 		hRes = CONVERT10_E_OLESTREAM_FMT;
9662 	}
9663 		else
9664 		{
9665 			hRes = S_OK;
9666 			break;
9667 		}
9668 	}
9669 
9670 	if(hRes == S_OK)
9671 	{
9672 		/* Get the TypeID... more info needed for this field */
9673 		dwSize = pOleStream->lpstbl->Get(pOleStream, (void *)&(pData->dwTypeID), sizeof(pData->dwTypeID));
9674 		if(dwSize != sizeof(pData->dwTypeID))
9675 		{
9676 			hRes = CONVERT10_E_OLESTREAM_GET;
9677 		}
9678 	}
9679 	if(hRes == S_OK)
9680 	{
9681 		if(pData->dwTypeID != 0)
9682 		{
9683 			/* Get the length of the OleTypeName */
9684 			dwSize = pOleStream->lpstbl->Get(pOleStream, (void *) &(pData->dwOleTypeNameLength), sizeof(pData->dwOleTypeNameLength));
9685 			if(dwSize != sizeof(pData->dwOleTypeNameLength))
9686 			{
9687 				hRes = CONVERT10_E_OLESTREAM_GET;
9688 			}
9689 
9690 			if(hRes == S_OK)
9691 			{
9692 				if(pData->dwOleTypeNameLength > 0)
9693 				{
9694 					/* Get the OleTypeName */
9695 					dwSize = pOleStream->lpstbl->Get(pOleStream, pData->strOleTypeName, pData->dwOleTypeNameLength);
9696 					if(dwSize != pData->dwOleTypeNameLength)
9697 					{
9698 						hRes = CONVERT10_E_OLESTREAM_GET;
9699 					}
9700 				}
9701 			}
9702 			if(bStrem1)
9703 			{
9704 				dwSize = pOleStream->lpstbl->Get(pOleStream, (void *)&(pData->dwOleObjFileNameLength), sizeof(pData->dwOleObjFileNameLength));
9705 				if(dwSize != sizeof(pData->dwOleObjFileNameLength))
9706 				{
9707 					hRes = CONVERT10_E_OLESTREAM_GET;
9708 				}
9709 			if(hRes == S_OK)
9710 			{
9711 					if(pData->dwOleObjFileNameLength < 1) /* there is no file name exist */
9712 						pData->dwOleObjFileNameLength = sizeof(pData->dwOleObjFileNameLength);
9713 					pData->pstrOleObjFileName = HeapAlloc(GetProcessHeap(), 0, pData->dwOleObjFileNameLength);
9714 					if(pData->pstrOleObjFileName)
9715 					{
9716 						dwSize = pOleStream->lpstbl->Get(pOleStream, pData->pstrOleObjFileName, pData->dwOleObjFileNameLength);
9717 						if(dwSize != pData->dwOleObjFileNameLength)
9718 						{
9719 							hRes = CONVERT10_E_OLESTREAM_GET;
9720 						}
9721 					}
9722 					else
9723 						hRes = CONVERT10_E_OLESTREAM_GET;
9724 				}
9725 			}
9726 			else
9727 			{
9728 				/* Get the Width of the Metafile */
9729 				dwSize = pOleStream->lpstbl->Get(pOleStream, (void *)&(pData->dwMetaFileWidth), sizeof(pData->dwMetaFileWidth));
9730 				if(dwSize != sizeof(pData->dwMetaFileWidth))
9731 				{
9732 					hRes = CONVERT10_E_OLESTREAM_GET;
9733 				}
9734 			if(hRes == S_OK)
9735 			{
9736 				/* Get the Height of the Metafile */
9737 				dwSize = pOleStream->lpstbl->Get(pOleStream, (void *)&(pData->dwMetaFileHeight), sizeof(pData->dwMetaFileHeight));
9738 				if(dwSize != sizeof(pData->dwMetaFileHeight))
9739 				{
9740 					hRes = CONVERT10_E_OLESTREAM_GET;
9741 				}
9742 			}
9743 			}
9744 			if(hRes == S_OK)
9745 			{
9746 				/* Get the Length of the Data */
9747 				dwSize = pOleStream->lpstbl->Get(pOleStream, (void *)&(pData->dwDataLength), sizeof(pData->dwDataLength));
9748 				if(dwSize != sizeof(pData->dwDataLength))
9749 				{
9750 					hRes = CONVERT10_E_OLESTREAM_GET;
9751 				}
9752 			}
9753 
9754 			if(hRes == S_OK) /* I don't know what this 8 byte information is. We have to figure out */
9755 			{
9756 				if(!bStrem1) /* if it is a second OLE stream data */
9757 				{
9758 					pData->dwDataLength -= 8;
9759 					dwSize = pOleStream->lpstbl->Get(pOleStream, pData->strUnknown, sizeof(pData->strUnknown));
9760 					if(dwSize != sizeof(pData->strUnknown))
9761 					{
9762 						hRes = CONVERT10_E_OLESTREAM_GET;
9763 					}
9764 				}
9765 			}
9766 			if(hRes == S_OK)
9767 			{
9768 				if(pData->dwDataLength > 0)
9769 				{
9770 					pData->pData = HeapAlloc(GetProcessHeap(),0,pData->dwDataLength);
9771 
9772 					/* Get Data (ex. IStorage, Metafile, or BMP) */
9773 					if(pData->pData)
9774 					{
9775 						dwSize = pOleStream->lpstbl->Get(pOleStream, (void *)pData->pData, pData->dwDataLength);
9776 						if(dwSize != pData->dwDataLength)
9777 						{
9778 							hRes = CONVERT10_E_OLESTREAM_GET;
9779 						}
9780 					}
9781 					else
9782 					{
9783 						hRes = CONVERT10_E_OLESTREAM_GET;
9784 					}
9785 				}
9786 			}
9787 		}
9788 	}
9789 	return hRes;
9790 }
9791 
9792 /*************************************************************************
9793  * OLECONVERT_SaveOLE10 [Internal]
9794  *
9795  * Saves the OLE10 STREAM From memory
9796  *
9797  * PARAMS
9798  *     pData        [I] Data Structure for the OLESTREAM Data
9799  *     pOleStream   [I] The OLESTREAM to save
9800  *
9801  * RETURNS
9802  *     Success:  S_OK
9803  *     Failure:  CONVERT10_E_OLESTREAM_PUT for invalid Put
9804  *
9805  * NOTES
9806  *     This function is used by OleConvertIStorageToOLESTREAM only.
9807  *
9808  */
9809 static HRESULT OLECONVERT_SaveOLE10(OLECONVERT_OLESTREAM_DATA *pData, LPOLESTREAM pOleStream)
9810 {
9811     DWORD dwSize;
9812     HRESULT hRes = S_OK;
9813 
9814 
9815    /* Set the OleID */
9816     dwSize = pOleStream->lpstbl->Put(pOleStream, (void *)&(pData->dwOleID), sizeof(pData->dwOleID));
9817     if(dwSize != sizeof(pData->dwOleID))
9818     {
9819         hRes = CONVERT10_E_OLESTREAM_PUT;
9820     }
9821 
9822     if(hRes == S_OK)
9823     {
9824         /* Set the TypeID */
9825         dwSize = pOleStream->lpstbl->Put(pOleStream, (void *)&(pData->dwTypeID), sizeof(pData->dwTypeID));
9826         if(dwSize != sizeof(pData->dwTypeID))
9827         {
9828             hRes = CONVERT10_E_OLESTREAM_PUT;
9829         }
9830     }
9831 
9832     if(pData->dwOleID == OLESTREAM_ID && pData->dwTypeID != 0 && hRes == S_OK)
9833     {
9834         /* Set the Length of the OleTypeName */
9835         dwSize = pOleStream->lpstbl->Put(pOleStream, (void *)&(pData->dwOleTypeNameLength), sizeof(pData->dwOleTypeNameLength));
9836         if(dwSize != sizeof(pData->dwOleTypeNameLength))
9837         {
9838             hRes = CONVERT10_E_OLESTREAM_PUT;
9839         }
9840 
9841         if(hRes == S_OK)
9842         {
9843             if(pData->dwOleTypeNameLength > 0)
9844             {
9845                 /* Set the OleTypeName */
9846                 dwSize = pOleStream->lpstbl->Put(pOleStream, pData->strOleTypeName, pData->dwOleTypeNameLength);
9847                 if(dwSize != pData->dwOleTypeNameLength)
9848                 {
9849                     hRes = CONVERT10_E_OLESTREAM_PUT;
9850                 }
9851             }
9852         }
9853 
9854         if(hRes == S_OK)
9855         {
9856             /* Set the width of the Metafile */
9857             dwSize = pOleStream->lpstbl->Put(pOleStream, (void *)&(pData->dwMetaFileWidth), sizeof(pData->dwMetaFileWidth));
9858             if(dwSize != sizeof(pData->dwMetaFileWidth))
9859             {
9860                 hRes = CONVERT10_E_OLESTREAM_PUT;
9861             }
9862         }
9863 
9864         if(hRes == S_OK)
9865         {
9866             /* Set the height of the Metafile */
9867             dwSize = pOleStream->lpstbl->Put(pOleStream, (void *)&(pData->dwMetaFileHeight), sizeof(pData->dwMetaFileHeight));
9868             if(dwSize != sizeof(pData->dwMetaFileHeight))
9869             {
9870                 hRes = CONVERT10_E_OLESTREAM_PUT;
9871             }
9872         }
9873 
9874         if(hRes == S_OK)
9875         {
9876             /* Set the length of the Data */
9877             dwSize = pOleStream->lpstbl->Put(pOleStream, (void *)&(pData->dwDataLength), sizeof(pData->dwDataLength));
9878             if(dwSize != sizeof(pData->dwDataLength))
9879             {
9880                 hRes = CONVERT10_E_OLESTREAM_PUT;
9881             }
9882         }
9883 
9884         if(hRes == S_OK)
9885         {
9886             if(pData->dwDataLength > 0)
9887             {
9888                 /* Set the Data (eg. IStorage, Metafile, Bitmap) */
9889                 dwSize = pOleStream->lpstbl->Put(pOleStream, (void *)  pData->pData, pData->dwDataLength);
9890                 if(dwSize != pData->dwDataLength)
9891                 {
9892                     hRes = CONVERT10_E_OLESTREAM_PUT;
9893                 }
9894             }
9895         }
9896     }
9897     return hRes;
9898 }
9899 
9900 /*************************************************************************
9901  * OLECONVERT_GetOLE20FromOLE10[Internal]
9902  *
9903  * This function copies OLE10 Data (the IStorage in the OLESTREAM) to disk,
9904  * opens it, and copies the content to the dest IStorage for
9905  * OleConvertOLESTREAMToIStorage
9906  *
9907  *
9908  * PARAMS
9909  *     pDestStorage  [I] The IStorage to copy the data to
9910  *     pBuffer       [I] Buffer that contains the IStorage from the OLESTREAM
9911  *     nBufferLength [I] The size of the buffer
9912  *
9913  * RETURNS
9914  *     Nothing
9915  *
9916  * NOTES
9917  *
9918  *
9919  */
9920 static void OLECONVERT_GetOLE20FromOLE10(LPSTORAGE pDestStorage, const BYTE *pBuffer, DWORD nBufferLength)
9921 {
9922     HRESULT hRes;
9923     HANDLE hFile;
9924     IStorage *pTempStorage;
9925     DWORD dwNumOfBytesWritten;
9926     WCHAR wstrTempDir[MAX_PATH], wstrTempFile[MAX_PATH];
9927     static const WCHAR wstrPrefix[] = {'s', 'i', 's', 0};
9928 
9929     /* Create a temp File */
9930     GetTempPathW(MAX_PATH, wstrTempDir);
9931     GetTempFileNameW(wstrTempDir, wstrPrefix, 0, wstrTempFile);
9932     hFile = CreateFileW(wstrTempFile, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
9933 
9934     if(hFile != INVALID_HANDLE_VALUE)
9935     {
9936         /* Write IStorage Data to File */
9937         WriteFile(hFile, pBuffer, nBufferLength, &dwNumOfBytesWritten, NULL);
9938         CloseHandle(hFile);
9939 
9940         /* Open and copy temp storage to the Dest Storage */
9941         hRes = StgOpenStorage(wstrTempFile, NULL, STGM_READ, NULL, 0, &pTempStorage);
9942         if(hRes == S_OK)
9943         {
9944             hRes = IStorage_CopyTo(pTempStorage, 0, NULL, NULL, pDestStorage);
9945             IStorage_Release(pTempStorage);
9946         }
9947         DeleteFileW(wstrTempFile);
9948     }
9949 }
9950 
9951 
9952 /*************************************************************************
9953  * OLECONVERT_WriteOLE20ToBuffer [Internal]
9954  *
9955  * Saves the OLE10 STREAM From memory
9956  *
9957  * PARAMS
9958  *     pStorage  [I] The Src IStorage to copy
9959  *     pData     [I] The Dest Memory to write to.
9960  *
9961  * RETURNS
9962  *     The size in bytes allocated for pData
9963  *
9964  * NOTES
9965  *     Memory allocated for pData must be freed by the caller
9966  *
9967  *     Used by OleConvertIStorageToOLESTREAM only.
9968  *
9969  */
9970 static DWORD OLECONVERT_WriteOLE20ToBuffer(LPSTORAGE pStorage, BYTE **pData)
9971 {
9972     HANDLE hFile;
9973     HRESULT hRes;
9974     DWORD nDataLength = 0;
9975     IStorage *pTempStorage;
9976     WCHAR wstrTempDir[MAX_PATH], wstrTempFile[MAX_PATH];
9977     static const WCHAR wstrPrefix[] = {'s', 'i', 's', 0};
9978 
9979     *pData = NULL;
9980 
9981     /* Create temp Storage */
9982     GetTempPathW(MAX_PATH, wstrTempDir);
9983     GetTempFileNameW(wstrTempDir, wstrPrefix, 0, wstrTempFile);
9984     hRes = StgCreateDocfile(wstrTempFile, STGM_CREATE | STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &pTempStorage);
9985 
9986     if(hRes == S_OK)
9987     {
9988         /* Copy Src Storage to the Temp Storage */
9989         IStorage_CopyTo(pStorage, 0, NULL, NULL, pTempStorage);
9990         IStorage_Release(pTempStorage);
9991 
9992         /* Open Temp Storage as a file and copy to memory */
9993         hFile = CreateFileW(wstrTempFile, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
9994         if(hFile != INVALID_HANDLE_VALUE)
9995         {
9996             nDataLength = GetFileSize(hFile, NULL);
9997             *pData = HeapAlloc(GetProcessHeap(),0,nDataLength);
9998             ReadFile(hFile, *pData, nDataLength, &nDataLength, 0);
9999             CloseHandle(hFile);
10000         }
10001         DeleteFileW(wstrTempFile);
10002     }
10003     return nDataLength;
10004 }
10005 
10006 /*************************************************************************
10007  * OLECONVERT_CreateCompObjStream [Internal]
10008  *
10009  * Creates a "\001CompObj" is the destination IStorage if necessary.
10010  *
10011  * PARAMS
10012  *     pStorage       [I] The dest IStorage to create the CompObj Stream
10013  *                        if necessary.
10014  *     strOleTypeName [I] The ProgID
10015  *
10016  * RETURNS
10017  *     Success:  S_OK
10018  *     Failure:  REGDB_E_CLASSNOTREG if cannot reconstruct the stream
10019  *
10020  * NOTES
10021  *     This function is used by OleConvertOLESTREAMToIStorage only.
10022  *
10023  *     The stream data is stored in the OLESTREAM and there should be
10024  *     no need to recreate the stream.  If the stream is manually
10025  *     deleted it will attempt to create it by querying the registry.
10026  *
10027  *
10028  */
10029 HRESULT OLECONVERT_CreateCompObjStream(LPSTORAGE pStorage, LPCSTR strOleTypeName)
10030 {
10031     IStream *pStream;
10032     HRESULT hStorageRes, hRes = S_OK;
10033     OLECONVERT_ISTORAGE_COMPOBJ IStorageCompObj;
10034     static const WCHAR wstrStreamName[] = {1,'C', 'o', 'm', 'p', 'O', 'b', 'j', 0};
10035     WCHAR bufferW[OLESTREAM_MAX_STR_LEN];
10036 
10037     static const BYTE pCompObjUnknown1[] = {0x01, 0x00, 0xFE, 0xFF, 0x03, 0x0A, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF};
10038     static const BYTE pCompObjUnknown2[] = {0xF4, 0x39, 0xB2, 0x71};
10039 
10040     /* Initialize the CompObj structure */
10041     memset(&IStorageCompObj, 0, sizeof(IStorageCompObj));
10042     memcpy(IStorageCompObj.byUnknown1, pCompObjUnknown1, sizeof(pCompObjUnknown1));
10043     memcpy(IStorageCompObj.byUnknown2, pCompObjUnknown2, sizeof(pCompObjUnknown2));
10044 
10045 
10046     /*  Create a CompObj stream if it doesn't exist */
10047     hStorageRes = IStorage_CreateStream(pStorage, wstrStreamName,
10048         STGM_WRITE  | STGM_SHARE_EXCLUSIVE, 0, 0, &pStream );
10049     if(hStorageRes == S_OK)
10050     {
10051         /* copy the OleTypeName to the compobj struct */
10052         IStorageCompObj.dwOleTypeNameLength = strlen(strOleTypeName)+1;
10053         strcpy(IStorageCompObj.strOleTypeName, strOleTypeName);
10054 
10055         /* copy the OleTypeName to the compobj struct */
10056         /* Note: in the test made, these were Identical      */
10057         IStorageCompObj.dwProgIDNameLength = strlen(strOleTypeName)+1;
10058         strcpy(IStorageCompObj.strProgIDName, strOleTypeName);
10059 
10060         /* Get the CLSID */
10061         MultiByteToWideChar( CP_ACP, 0, IStorageCompObj.strProgIDName, -1,
10062                              bufferW, OLESTREAM_MAX_STR_LEN );
10063         hRes = CLSIDFromProgID(bufferW, &(IStorageCompObj.clsid));
10064 
10065         if(hRes == S_OK)
10066         {
10067             HKEY hKey;
10068             LONG hErr;
10069             /* Get the CLSID Default Name from the Registry */
10070             hErr = open_classes_key(HKEY_CLASSES_ROOT, bufferW, MAXIMUM_ALLOWED, &hKey);
10071             if(hErr == ERROR_SUCCESS)
10072             {
10073                 char strTemp[OLESTREAM_MAX_STR_LEN];
10074                 IStorageCompObj.dwCLSIDNameLength = OLESTREAM_MAX_STR_LEN;
10075                 hErr = RegQueryValueA(hKey, NULL, strTemp, (LONG*) &(IStorageCompObj.dwCLSIDNameLength));
10076                 if(hErr == ERROR_SUCCESS)
10077                 {
10078                     strcpy(IStorageCompObj.strCLSIDName, strTemp);
10079                 }
10080                 RegCloseKey(hKey);
10081             }
10082         }
10083 
10084         /* Write CompObj Structure to stream */
10085         hRes = IStream_Write(pStream, IStorageCompObj.byUnknown1, sizeof(IStorageCompObj.byUnknown1), NULL);
10086 
10087         WriteClassStm(pStream,&(IStorageCompObj.clsid));
10088 
10089         hRes = IStream_Write(pStream, &(IStorageCompObj.dwCLSIDNameLength), sizeof(IStorageCompObj.dwCLSIDNameLength), NULL);
10090         if(IStorageCompObj.dwCLSIDNameLength > 0)
10091         {
10092             hRes = IStream_Write(pStream, IStorageCompObj.strCLSIDName, IStorageCompObj.dwCLSIDNameLength, NULL);
10093         }
10094         hRes = IStream_Write(pStream, &(IStorageCompObj.dwOleTypeNameLength) , sizeof(IStorageCompObj.dwOleTypeNameLength), NULL);
10095         if(IStorageCompObj.dwOleTypeNameLength > 0)
10096         {
10097             hRes = IStream_Write(pStream, IStorageCompObj.strOleTypeName , IStorageCompObj.dwOleTypeNameLength, NULL);
10098         }
10099         hRes = IStream_Write(pStream, &(IStorageCompObj.dwProgIDNameLength) , sizeof(IStorageCompObj.dwProgIDNameLength), NULL);
10100         if(IStorageCompObj.dwProgIDNameLength > 0)
10101         {
10102             hRes = IStream_Write(pStream, IStorageCompObj.strProgIDName , IStorageCompObj.dwProgIDNameLength, NULL);
10103         }
10104         hRes = IStream_Write(pStream, IStorageCompObj.byUnknown2 , sizeof(IStorageCompObj.byUnknown2), NULL);
10105         IStream_Release(pStream);
10106     }
10107     return hRes;
10108 }
10109 
10110 
10111 /*************************************************************************
10112  * OLECONVERT_CreateOlePresStream[Internal]
10113  *
10114  * Creates the "\002OlePres000" Stream with the Metafile data
10115  *
10116  * PARAMS
10117  *     pStorage     [I] The dest IStorage to create \002OLEPres000 stream in.
10118  *     dwExtentX    [I] Width of the Metafile
10119  *     dwExtentY    [I] Height of the Metafile
10120  *     pData        [I] Metafile data
10121  *     dwDataLength [I] Size of the Metafile data
10122  *
10123  * RETURNS
10124  *     Success:  S_OK
10125  *     Failure:  CONVERT10_E_OLESTREAM_PUT for invalid Put
10126  *
10127  * NOTES
10128  *     This function is used by OleConvertOLESTREAMToIStorage only.
10129  *
10130  */
10131 static void OLECONVERT_CreateOlePresStream(LPSTORAGE pStorage, DWORD dwExtentX, DWORD dwExtentY , BYTE *pData, DWORD dwDataLength)
10132 {
10133     HRESULT hRes;
10134     IStream *pStream;
10135     static const WCHAR wstrStreamName[] = {2, 'O', 'l', 'e', 'P', 'r', 'e', 's', '0', '0', '0', 0};
10136     static const BYTE pOlePresStreamHeader[] =
10137     {
10138         0xFF, 0xFF, 0xFF, 0xFF, 0x03, 0x00, 0x00, 0x00,
10139         0x04, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00,
10140         0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00,
10141         0x00, 0x00, 0x00, 0x00
10142     };
10143 
10144     static const BYTE pOlePresStreamHeaderEmpty[] =
10145     {
10146         0x00, 0x00, 0x00, 0x00,
10147         0x04, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00,
10148         0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00,
10149         0x00, 0x00, 0x00, 0x00
10150     };
10151 
10152     /* Create the OlePres000 Stream */
10153     hRes = IStorage_CreateStream(pStorage, wstrStreamName,
10154         STGM_CREATE | STGM_WRITE  | STGM_SHARE_EXCLUSIVE, 0, 0, &pStream );
10155 
10156     if(hRes == S_OK)
10157     {
10158         DWORD nHeaderSize;
10159         OLECONVERT_ISTORAGE_OLEPRES OlePres;
10160 
10161         memset(&OlePres, 0, sizeof(OlePres));
10162         /* Do we have any metafile data to save */
10163         if(dwDataLength > 0)
10164         {
10165             memcpy(OlePres.byUnknown1, pOlePresStreamHeader, sizeof(pOlePresStreamHeader));
10166             nHeaderSize = sizeof(pOlePresStreamHeader);
10167         }
10168         else
10169         {
10170             memcpy(OlePres.byUnknown1, pOlePresStreamHeaderEmpty, sizeof(pOlePresStreamHeaderEmpty));
10171             nHeaderSize = sizeof(pOlePresStreamHeaderEmpty);
10172         }
10173         /* Set width and height of the metafile */
10174         OlePres.dwExtentX = dwExtentX;
10175         OlePres.dwExtentY = -dwExtentY;
10176 
10177         /* Set Data and Length */
10178         if(dwDataLength > sizeof(METAFILEPICT16))
10179         {
10180             OlePres.dwSize = dwDataLength - sizeof(METAFILEPICT16);
10181             OlePres.pData = &(pData[8]);
10182         }
10183         /* Save OlePres000 Data to Stream */
10184         hRes = IStream_Write(pStream, OlePres.byUnknown1, nHeaderSize, NULL);
10185         hRes = IStream_Write(pStream, &(OlePres.dwExtentX), sizeof(OlePres.dwExtentX), NULL);
10186         hRes = IStream_Write(pStream, &(OlePres.dwExtentY), sizeof(OlePres.dwExtentY), NULL);
10187         hRes = IStream_Write(pStream, &(OlePres.dwSize), sizeof(OlePres.dwSize), NULL);
10188         if(OlePres.dwSize > 0)
10189         {
10190             hRes = IStream_Write(pStream, OlePres.pData, OlePres.dwSize, NULL);
10191         }
10192         IStream_Release(pStream);
10193     }
10194 }
10195 
10196 /*************************************************************************
10197  * OLECONVERT_CreateOle10NativeStream [Internal]
10198  *
10199  * Creates the "\001Ole10Native" Stream (should contain a BMP)
10200  *
10201  * PARAMS
10202  *     pStorage     [I] Dest storage to create the stream in
10203  *     pData        [I] Ole10 Native Data (ex. bmp)
10204  *     dwDataLength [I] Size of the Ole10 Native Data
10205  *
10206  * RETURNS
10207  *     Nothing
10208  *
10209  * NOTES
10210  *     This function is used by OleConvertOLESTREAMToIStorage only.
10211  *
10212  *     Might need to verify the data and return appropriate error message
10213  *
10214  */
10215 static void OLECONVERT_CreateOle10NativeStream(LPSTORAGE pStorage, const BYTE *pData, DWORD dwDataLength)
10216 {
10217     HRESULT hRes;
10218     IStream *pStream;
10219     static const WCHAR wstrStreamName[] = {1, 'O', 'l', 'e', '1', '0', 'N', 'a', 't', 'i', 'v', 'e', 0};
10220 
10221     /* Create the Ole10Native Stream */
10222     hRes = IStorage_CreateStream(pStorage, wstrStreamName,
10223         STGM_CREATE | STGM_WRITE  | STGM_SHARE_EXCLUSIVE, 0, 0, &pStream );
10224 
10225     if(hRes == S_OK)
10226     {
10227         /* Write info to stream */
10228         hRes = IStream_Write(pStream, &dwDataLength, sizeof(dwDataLength), NULL);
10229         hRes = IStream_Write(pStream, pData, dwDataLength, NULL);
10230         IStream_Release(pStream);
10231     }
10232 
10233 }
10234 
10235 /*************************************************************************
10236  * OLECONVERT_GetOLE10ProgID [Internal]
10237  *
10238  * Finds the ProgID (or OleTypeID) from the IStorage
10239  *
10240  * PARAMS
10241  *     pStorage        [I] The Src IStorage to get the ProgID
10242  *     strProgID       [I] the ProgID string to get
10243  *     dwSize          [I] the size of the string
10244  *
10245  * RETURNS
10246  *     Success:  S_OK
10247  *     Failure:  REGDB_E_CLASSNOTREG if cannot reconstruct the stream
10248  *
10249  * NOTES
10250  *     This function is used by OleConvertIStorageToOLESTREAM only.
10251  *
10252  *
10253  */
10254 static HRESULT OLECONVERT_GetOLE10ProgID(LPSTORAGE pStorage, char *strProgID, DWORD *dwSize)
10255 {
10256     HRESULT hRes;
10257     IStream *pStream;
10258     LARGE_INTEGER iSeekPos;
10259     OLECONVERT_ISTORAGE_COMPOBJ CompObj;
10260     static const WCHAR wstrStreamName[] = {1,'C', 'o', 'm', 'p', 'O', 'b', 'j', 0};
10261 
10262     /* Open the CompObj Stream */
10263     hRes = IStorage_OpenStream(pStorage, wstrStreamName, NULL,
10264         STGM_READ  | STGM_SHARE_EXCLUSIVE, 0, &pStream );
10265     if(hRes == S_OK)
10266     {
10267 
10268         /*Get the OleType from the CompObj Stream */
10269         iSeekPos.u.LowPart = sizeof(CompObj.byUnknown1) + sizeof(CompObj.clsid);
10270         iSeekPos.u.HighPart = 0;
10271 
10272         IStream_Seek(pStream, iSeekPos, STREAM_SEEK_SET, NULL);
10273         IStream_Read(pStream, &CompObj.dwCLSIDNameLength, sizeof(CompObj.dwCLSIDNameLength), NULL);
10274         iSeekPos.u.LowPart = CompObj.dwCLSIDNameLength;
10275         IStream_Seek(pStream, iSeekPos, STREAM_SEEK_CUR , NULL);
10276         IStream_Read(pStream, &CompObj.dwOleTypeNameLength, sizeof(CompObj.dwOleTypeNameLength), NULL);
10277         iSeekPos.u.LowPart = CompObj.dwOleTypeNameLength;
10278         IStream_Seek(pStream, iSeekPos, STREAM_SEEK_CUR , NULL);
10279 
10280         IStream_Read(pStream, dwSize, sizeof(*dwSize), NULL);
10281         if(*dwSize > 0)
10282         {
10283             IStream_Read(pStream, strProgID, *dwSize, NULL);
10284         }
10285         IStream_Release(pStream);
10286     }
10287     else
10288     {
10289         STATSTG stat;
10290         LPOLESTR wstrProgID;
10291 
10292         /* Get the OleType from the registry */
10293         REFCLSID clsid = &(stat.clsid);
10294         IStorage_Stat(pStorage, &stat, STATFLAG_NONAME);
10295         hRes = ProgIDFromCLSID(clsid, &wstrProgID);
10296         if(hRes == S_OK)
10297         {
10298             *dwSize = WideCharToMultiByte(CP_ACP, 0, wstrProgID, -1, strProgID, *dwSize, NULL, FALSE);
10299             CoTaskMemFree(wstrProgID);
10300         }
10301 
10302     }
10303     return hRes;
10304 }
10305 
10306 /*************************************************************************
10307  * OLECONVERT_GetOle10PresData [Internal]
10308  *
10309  * Converts IStorage "/001Ole10Native" stream to a OLE10 Stream
10310  *
10311  * PARAMS
10312  *     pStorage     [I] Src IStroage
10313  *     pOleStream   [I] Dest OleStream Mem Struct
10314  *
10315  * RETURNS
10316  *     Nothing
10317  *
10318  * NOTES
10319  *     This function is used by OleConvertIStorageToOLESTREAM only.
10320  *
10321  *     Memory allocated for pData must be freed by the caller
10322  *
10323  *
10324  */
10325 static void OLECONVERT_GetOle10PresData(LPSTORAGE pStorage, OLECONVERT_OLESTREAM_DATA *pOleStreamData)
10326 {
10327 
10328     HRESULT hRes;
10329     IStream *pStream;
10330     static const WCHAR wstrStreamName[] = {1, 'O', 'l', 'e', '1', '0', 'N', 'a', 't', 'i', 'v', 'e', 0};
10331 
10332     /* Initialize Default data for OLESTREAM */
10333     pOleStreamData[0].dwOleID = OLESTREAM_ID;
10334     pOleStreamData[0].dwTypeID = 2;
10335     pOleStreamData[1].dwOleID = OLESTREAM_ID;
10336     pOleStreamData[1].dwTypeID = 0;
10337     pOleStreamData[0].dwMetaFileWidth = 0;
10338     pOleStreamData[0].dwMetaFileHeight = 0;
10339     pOleStreamData[0].pData = NULL;
10340     pOleStreamData[1].pData = NULL;
10341 
10342     /* Open Ole10Native Stream */
10343     hRes = IStorage_OpenStream(pStorage, wstrStreamName, NULL,
10344         STGM_READ  | STGM_SHARE_EXCLUSIVE, 0, &pStream );
10345     if(hRes == S_OK)
10346     {
10347 
10348         /* Read Size and Data */
10349         IStream_Read(pStream, &(pOleStreamData->dwDataLength), sizeof(pOleStreamData->dwDataLength), NULL);
10350         if(pOleStreamData->dwDataLength > 0)
10351         {
10352             pOleStreamData->pData = HeapAlloc(GetProcessHeap(),0,pOleStreamData->dwDataLength);
10353             IStream_Read(pStream, pOleStreamData->pData, pOleStreamData->dwDataLength, NULL);
10354         }
10355         IStream_Release(pStream);
10356     }
10357 
10358 }
10359 
10360 
10361 /*************************************************************************
10362  * OLECONVERT_GetOle20PresData[Internal]
10363  *
10364  * Converts IStorage "/002OlePres000" stream to a OLE10 Stream
10365  *
10366  * PARAMS
10367  *     pStorage         [I] Src IStroage
10368  *     pOleStreamData   [I] Dest OleStream Mem Struct
10369  *
10370  * RETURNS
10371  *     Nothing
10372  *
10373  * NOTES
10374  *     This function is used by OleConvertIStorageToOLESTREAM only.
10375  *
10376  *     Memory allocated for pData must be freed by the caller
10377  */
10378 static void OLECONVERT_GetOle20PresData(LPSTORAGE pStorage, OLECONVERT_OLESTREAM_DATA *pOleStreamData)
10379 {
10380     HRESULT hRes;
10381     IStream *pStream;
10382     OLECONVERT_ISTORAGE_OLEPRES olePress;
10383     static const WCHAR wstrStreamName[] = {2, 'O', 'l', 'e', 'P', 'r', 'e', 's', '0', '0', '0', 0};
10384 
10385     /* Initialize Default data for OLESTREAM */
10386     pOleStreamData[0].dwOleID = OLESTREAM_ID;
10387     pOleStreamData[0].dwTypeID = 2;
10388     pOleStreamData[0].dwMetaFileWidth = 0;
10389     pOleStreamData[0].dwMetaFileHeight = 0;
10390     pOleStreamData[0].dwDataLength = OLECONVERT_WriteOLE20ToBuffer(pStorage, &(pOleStreamData[0].pData));
10391     pOleStreamData[1].dwOleID = OLESTREAM_ID;
10392     pOleStreamData[1].dwTypeID = 0;
10393     pOleStreamData[1].dwOleTypeNameLength = 0;
10394     pOleStreamData[1].strOleTypeName[0] = 0;
10395     pOleStreamData[1].dwMetaFileWidth = 0;
10396     pOleStreamData[1].dwMetaFileHeight = 0;
10397     pOleStreamData[1].pData = NULL;
10398     pOleStreamData[1].dwDataLength = 0;
10399 
10400 
10401     /* Open OlePress000 stream */
10402     hRes = IStorage_OpenStream(pStorage, wstrStreamName, NULL,
10403         STGM_READ  | STGM_SHARE_EXCLUSIVE, 0, &pStream );
10404     if(hRes == S_OK)
10405     {
10406         LARGE_INTEGER iSeekPos;
10407         METAFILEPICT16 MetaFilePict;
10408         static const char strMetafilePictName[] = "METAFILEPICT";
10409 
10410         /* Set the TypeID for a Metafile */
10411         pOleStreamData[1].dwTypeID = 5;
10412 
10413         /* Set the OleTypeName to Metafile */
10414         pOleStreamData[1].dwOleTypeNameLength = strlen(strMetafilePictName) +1;
10415         strcpy(pOleStreamData[1].strOleTypeName, strMetafilePictName);
10416 
10417         iSeekPos.u.HighPart = 0;
10418         iSeekPos.u.LowPart = sizeof(olePress.byUnknown1);
10419 
10420         /* Get Presentation Data */
10421         IStream_Seek(pStream, iSeekPos, STREAM_SEEK_SET, NULL);
10422         IStream_Read(pStream, &(olePress.dwExtentX), sizeof(olePress.dwExtentX), NULL);
10423         IStream_Read(pStream, &(olePress.dwExtentY), sizeof(olePress.dwExtentY), NULL);
10424         IStream_Read(pStream, &(olePress.dwSize), sizeof(olePress.dwSize), NULL);
10425 
10426         /*Set width and Height */
10427         pOleStreamData[1].dwMetaFileWidth = olePress.dwExtentX;
10428         pOleStreamData[1].dwMetaFileHeight = -olePress.dwExtentY;
10429         if(olePress.dwSize > 0)
10430         {
10431             /* Set Length */
10432             pOleStreamData[1].dwDataLength  = olePress.dwSize + sizeof(METAFILEPICT16);
10433 
10434             /* Set MetaFilePict struct */
10435             MetaFilePict.mm = 8;
10436             MetaFilePict.xExt = olePress.dwExtentX;
10437             MetaFilePict.yExt = olePress.dwExtentY;
10438             MetaFilePict.hMF = 0;
10439 
10440             /* Get Metafile Data */
10441             pOleStreamData[1].pData = HeapAlloc(GetProcessHeap(),0,pOleStreamData[1].dwDataLength);
10442             memcpy(pOleStreamData[1].pData, &MetaFilePict, sizeof(MetaFilePict));
10443             IStream_Read(pStream, &(pOleStreamData[1].pData[sizeof(MetaFilePict)]), pOleStreamData[1].dwDataLength-sizeof(METAFILEPICT16), NULL);
10444         }
10445         IStream_Release(pStream);
10446     }
10447 }
10448 
10449 /*************************************************************************
10450  * OleConvertOLESTREAMToIStorage [OLE32.@]
10451  *
10452  * Read info on MSDN
10453  *
10454  * TODO
10455  *      DVTARGETDEVICE parameter is not handled
10456  *      Still unsure of some mem fields for OLE 10 Stream
10457  *      Still some unknowns for the IStorage: "\002OlePres000", "\001CompObj",
10458  *      and "\001OLE" streams
10459  *
10460  */
10461 HRESULT WINAPI OleConvertOLESTREAMToIStorage (
10462     LPOLESTREAM pOleStream,
10463     LPSTORAGE pstg,
10464     const DVTARGETDEVICE* ptd)
10465 {
10466     int i;
10467     HRESULT hRes=S_OK;
10468     OLECONVERT_OLESTREAM_DATA pOleStreamData[2];
10469 
10470     TRACE("%p %p %p\n", pOleStream, pstg, ptd);
10471 
10472     memset(pOleStreamData, 0, sizeof(pOleStreamData));
10473 
10474     if(ptd != NULL)
10475     {
10476         FIXME("DVTARGETDEVICE is not NULL, unhandled parameter\n");
10477     }
10478 
10479     if(pstg == NULL || pOleStream == NULL)
10480     {
10481         hRes = E_INVALIDARG;
10482     }
10483 
10484     if(hRes == S_OK)
10485     {
10486         /* Load the OLESTREAM to Memory */
10487         hRes = OLECONVERT_LoadOLE10(pOleStream, &pOleStreamData[0], TRUE);
10488     }
10489 
10490     if(hRes == S_OK)
10491     {
10492         /* Load the OLESTREAM to Memory (part 2)*/
10493         hRes = OLECONVERT_LoadOLE10(pOleStream, &pOleStreamData[1], FALSE);
10494     }
10495 
10496     if(hRes == S_OK)
10497     {
10498 
10499         if(pOleStreamData[0].dwDataLength > sizeof(STORAGE_magic))
10500         {
10501             /* Do we have the IStorage Data in the OLESTREAM */
10502             if(memcmp(pOleStreamData[0].pData, STORAGE_magic, sizeof(STORAGE_magic)) ==0)
10503             {
10504                 OLECONVERT_GetOLE20FromOLE10(pstg, pOleStreamData[0].pData, pOleStreamData[0].dwDataLength);
10505                 OLECONVERT_CreateOlePresStream(pstg, pOleStreamData[1].dwMetaFileWidth, pOleStreamData[1].dwMetaFileHeight, pOleStreamData[1].pData, pOleStreamData[1].dwDataLength);
10506             }
10507             else
10508             {
10509                 /* It must be an original OLE 1.0 source */
10510                 OLECONVERT_CreateOle10NativeStream(pstg, pOleStreamData[0].pData, pOleStreamData[0].dwDataLength);
10511             }
10512         }
10513         else
10514         {
10515             /* It must be an original OLE 1.0 source */
10516             OLECONVERT_CreateOle10NativeStream(pstg, pOleStreamData[0].pData, pOleStreamData[0].dwDataLength);
10517         }
10518 
10519         /* Create CompObj Stream if necessary */
10520         hRes = OLECONVERT_CreateCompObjStream(pstg, pOleStreamData[0].strOleTypeName);
10521         if(hRes == S_OK)
10522         {
10523             /*Create the Ole Stream if necessary */
10524             STORAGE_CreateOleStream(pstg, 0);
10525         }
10526     }
10527 
10528 
10529     /* Free allocated memory */
10530     for(i=0; i < 2; i++)
10531     {
10532         HeapFree(GetProcessHeap(),0,pOleStreamData[i].pData);
10533         HeapFree(GetProcessHeap(),0,pOleStreamData[i].pstrOleObjFileName);
10534         pOleStreamData[i].pstrOleObjFileName = NULL;
10535     }
10536     return hRes;
10537 }
10538 
10539 /*************************************************************************
10540  * OleConvertIStorageToOLESTREAM [OLE32.@]
10541  *
10542  * Read info on MSDN
10543  *
10544  * Read info on MSDN
10545  *
10546  * TODO
10547  *      Still unsure of some mem fields for OLE 10 Stream
10548  *      Still some unknowns for the IStorage: "\002OlePres000", "\001CompObj",
10549  *      and "\001OLE" streams.
10550  *
10551  */
10552 HRESULT WINAPI OleConvertIStorageToOLESTREAM (
10553     LPSTORAGE pstg,
10554     LPOLESTREAM pOleStream)
10555 {
10556     int i;
10557     HRESULT hRes = S_OK;
10558     IStream *pStream;
10559     OLECONVERT_OLESTREAM_DATA pOleStreamData[2];
10560     static const WCHAR wstrStreamName[] = {1, 'O', 'l', 'e', '1', '0', 'N', 'a', 't', 'i', 'v', 'e', 0};
10561 
10562     TRACE("%p %p\n", pstg, pOleStream);
10563 
10564     memset(pOleStreamData, 0, sizeof(pOleStreamData));
10565 
10566     if(pstg == NULL || pOleStream == NULL)
10567     {
10568         hRes = E_INVALIDARG;
10569     }
10570     if(hRes == S_OK)
10571     {
10572         /* Get the ProgID */
10573         pOleStreamData[0].dwOleTypeNameLength = OLESTREAM_MAX_STR_LEN;
10574         hRes = OLECONVERT_GetOLE10ProgID(pstg, pOleStreamData[0].strOleTypeName, &(pOleStreamData[0].dwOleTypeNameLength));
10575     }
10576     if(hRes == S_OK)
10577     {
10578         /* Was it originally Ole10 */
10579         hRes = IStorage_OpenStream(pstg, wstrStreamName, 0, STGM_READ | STGM_SHARE_EXCLUSIVE, 0, &pStream);
10580         if(hRes == S_OK)
10581         {
10582             IStream_Release(pStream);
10583             /* Get Presentation Data for Ole10Native */
10584             OLECONVERT_GetOle10PresData(pstg, pOleStreamData);
10585         }
10586         else
10587         {
10588             /* Get Presentation Data (OLE20) */
10589             OLECONVERT_GetOle20PresData(pstg, pOleStreamData);
10590         }
10591 
10592         /* Save OLESTREAM */
10593         hRes = OLECONVERT_SaveOLE10(&(pOleStreamData[0]), pOleStream);
10594         if(hRes == S_OK)
10595         {
10596             hRes = OLECONVERT_SaveOLE10(&(pOleStreamData[1]), pOleStream);
10597         }
10598 
10599     }
10600 
10601     /* Free allocated memory */
10602     for(i=0; i < 2; i++)
10603     {
10604         HeapFree(GetProcessHeap(),0,pOleStreamData[i].pData);
10605     }
10606 
10607     return hRes;
10608 }
10609 
10610 enum stream_1ole_flags {
10611     OleStream_LinkedObject = 0x00000001,
10612     OleStream_Convert      = 0x00000004
10613 };
10614 
10615 /***********************************************************************
10616  *		GetConvertStg (OLE32.@)
10617  */
10618 HRESULT WINAPI GetConvertStg(IStorage *stg)
10619 {
10620     static const WCHAR stream_1oleW[] = {1,'O','l','e',0};
10621     static const DWORD version_magic = 0x02000001;
10622     DWORD header[2];
10623     IStream *stream;
10624     HRESULT hr;
10625 
10626     TRACE("%p\n", stg);
10627 
10628     if (!stg) return E_INVALIDARG;
10629 
10630     hr = IStorage_OpenStream(stg, stream_1oleW, NULL, STGM_READ | STGM_SHARE_EXCLUSIVE, 0, &stream);
10631     if (FAILED(hr)) return hr;
10632 
10633     hr = IStream_Read(stream, header, sizeof(header), NULL);
10634     IStream_Release(stream);
10635     if (FAILED(hr)) return hr;
10636 
10637     if (header[0] != version_magic)
10638     {
10639         ERR("got wrong version magic for 1Ole stream, 0x%08x\n", header[0]);
10640         return E_FAIL;
10641     }
10642 
10643     return header[1] & OleStream_Convert ? S_OK : S_FALSE;
10644 }
10645 
10646 /***********************************************************************
10647  *		SetConvertStg (OLE32.@)
10648  */
10649 HRESULT WINAPI SetConvertStg(IStorage *storage, BOOL convert)
10650 {
10651     static const WCHAR stream_1oleW[] = {1,'O','l','e',0};
10652     DWORD flags = convert ? OleStream_Convert : 0;
10653     IStream *stream;
10654     DWORD header[2];
10655     HRESULT hr;
10656 
10657     TRACE("(%p, %d)\n", storage, convert);
10658 
10659     hr = IStorage_OpenStream(storage, stream_1oleW, NULL, STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &stream);
10660     if (FAILED(hr))
10661     {
10662         if (hr != STG_E_FILENOTFOUND)
10663             return hr;
10664 
10665         return STORAGE_CreateOleStream(storage, flags);
10666     }
10667 
10668     hr = IStream_Read(stream, header, sizeof(header), NULL);
10669     if (FAILED(hr))
10670     {
10671         IStream_Release(stream);
10672         return hr;
10673     }
10674 
10675     /* update flag if differs */
10676     if ((header[1] ^ flags) & OleStream_Convert)
10677     {
10678         LARGE_INTEGER pos = {{0}};
10679 
10680         if (header[1] & OleStream_Convert)
10681             flags = header[1] & ~OleStream_Convert;
10682         else
10683             flags = header[1] |  OleStream_Convert;
10684 
10685         pos.QuadPart = sizeof(DWORD);
10686         hr = IStream_Seek(stream, pos, STREAM_SEEK_SET, NULL);
10687         if (FAILED(hr))
10688         {
10689             IStream_Release(stream);
10690             return hr;
10691         }
10692 
10693         hr = IStream_Write(stream, &flags, sizeof(flags), NULL);
10694     }
10695 
10696     IStream_Release(stream);
10697     return hr;
10698 }
10699