blob: 41320b8ec4dc816a71c88eebed53e9aaad3f34b7 [file] [log] [blame]
Francis Beaudetd92b9471999-01-30 13:24:23 +00001/*
2 * Compound Storage (32 bit version)
3 * Stream implementation
4 *
5 * This file contains the implementation of the stream interface
6 * for streams contained in a compound storage.
7 *
8 * Copyright 1999 Francis Beaudet
9 * Copyright 1999 Thuy Nguyen
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000010 *
11 * This library is free software; you can redistribute it and/or
12 * modify it under the terms of the GNU Lesser General Public
13 * License as published by the Free Software Foundation; either
14 * version 2.1 of the License, or (at your option) any later version.
15 *
16 * This library is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
19 * Lesser General Public License for more details.
20 *
21 * You should have received a copy of the GNU Lesser General Public
22 * License along with this library; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
Francis Beaudetd92b9471999-01-30 13:24:23 +000024 */
Dimitrie O. Paun297f3d82003-01-07 20:36:20 +000025
Francis Beaudetd92b9471999-01-30 13:24:23 +000026#include <assert.h>
27#include <stdlib.h>
28#include <stdio.h>
29#include <string.h>
30
Dimitrie O. Paun297f3d82003-01-07 20:36:20 +000031#define NONAMELESSUNION
32#define NONAMELESSSTRUCT
Marcus Meissner064f1701999-02-28 19:14:33 +000033#include "winbase.h"
Francis Beaudetd92b9471999-01-30 13:24:23 +000034#include "winerror.h"
Patrik Stridvall9c1de6d2002-09-12 22:07:02 +000035#include "winternl.h"
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000036#include "wine/debug.h"
Francis Beaudetd92b9471999-01-30 13:24:23 +000037
38#include "storage32.h"
39
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000040WINE_DEFAULT_DEBUG_CHANNEL(storage);
Patrik Stridvallb4b9fae1999-04-19 14:56:29 +000041
Francis Beaudetd92b9471999-01-30 13:24:23 +000042
43/*
44 * Virtual function table for the StgStreamImpl class.
45 */
Alexandre Julliarda3960291999-02-26 11:11:13 +000046static ICOM_VTABLE(IStream) StgStreamImpl_Vtbl =
Francis Beaudetd92b9471999-01-30 13:24:23 +000047{
Paul Quinn2305f3c1999-05-22 11:41:38 +000048 ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE
Francois Gougetb0c61291999-02-18 13:26:22 +000049 StgStreamImpl_QueryInterface,
50 StgStreamImpl_AddRef,
51 StgStreamImpl_Release,
52 StgStreamImpl_Read,
53 StgStreamImpl_Write,
54 StgStreamImpl_Seek,
55 StgStreamImpl_SetSize,
56 StgStreamImpl_CopyTo,
57 StgStreamImpl_Commit,
58 StgStreamImpl_Revert,
59 StgStreamImpl_LockRegion,
60 StgStreamImpl_UnlockRegion,
61 StgStreamImpl_Stat,
62 StgStreamImpl_Clone
Francis Beaudetd92b9471999-01-30 13:24:23 +000063};
64
65/******************************************************************************
66** StgStreamImpl implementation
67*/
68
69/***
70 * This is the constructor for the StgStreamImpl class.
71 *
72 * Params:
73 * parentStorage - Pointer to the storage that contains the stream to open
74 * ownerProperty - Index of the property that points to this stream.
75 */
76StgStreamImpl* StgStreamImpl_Construct(
Alexandre Julliarda3960291999-02-26 11:11:13 +000077 StorageBaseImpl* parentStorage,
Pierre Mageau89aa8612000-02-25 20:58:25 +000078 DWORD grfMode,
79 ULONG ownerProperty)
Francis Beaudetd92b9471999-01-30 13:24:23 +000080{
81 StgStreamImpl* newStream;
82
83 newStream = HeapAlloc(GetProcessHeap(), 0, sizeof(StgStreamImpl));
Vincent Béron9a624912002-05-31 23:06:46 +000084
Francis Beaudetd92b9471999-01-30 13:24:23 +000085 if (newStream!=0)
86 {
87 /*
88 * Set-up the virtual function table and reference count.
89 */
Francois Gouget01c9ac41999-10-31 01:59:23 +000090 ICOM_VTBL(newStream) = &StgStreamImpl_Vtbl;
91 newStream->ref = 0;
Vincent Béron9a624912002-05-31 23:06:46 +000092
Francis Beaudetd92b9471999-01-30 13:24:23 +000093 /*
94 * We want to nail-down the reference to the storage in case the
95 * stream out-lives the storage in the client application.
96 */
97 newStream->parentStorage = parentStorage;
Alexandre Julliarda3960291999-02-26 11:11:13 +000098 IStorage_AddRef((IStorage*)newStream->parentStorage);
Pierre Mageau89aa8612000-02-25 20:58:25 +000099
Vincent Béron9a624912002-05-31 23:06:46 +0000100 newStream->grfMode = grfMode;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000101 newStream->ownerProperty = ownerProperty;
Vincent Béron9a624912002-05-31 23:06:46 +0000102
Francis Beaudetd92b9471999-01-30 13:24:23 +0000103 /*
Francois Gouget3bb9a362001-10-22 19:04:32 +0000104 * Start the stream at the beginning.
Francis Beaudetd92b9471999-01-30 13:24:23 +0000105 */
Patrik Stridvall311e4561999-09-19 14:20:33 +0000106 newStream->currentPosition.s.HighPart = 0;
107 newStream->currentPosition.s.LowPart = 0;
Vincent Béron9a624912002-05-31 23:06:46 +0000108
Francis Beaudetd92b9471999-01-30 13:24:23 +0000109 /*
110 * Initialize the rest of the data.
111 */
Patrik Stridvall311e4561999-09-19 14:20:33 +0000112 newStream->streamSize.s.HighPart = 0;
113 newStream->streamSize.s.LowPart = 0;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000114 newStream->bigBlockChain = 0;
115 newStream->smallBlockChain = 0;
Vincent Béron9a624912002-05-31 23:06:46 +0000116
Francis Beaudetd92b9471999-01-30 13:24:23 +0000117 /*
118 * Read the size from the property and determine if the blocks forming
119 * this stream are large or small.
120 */
121 StgStreamImpl_OpenBlockChain(newStream);
122 }
Vincent Béron9a624912002-05-31 23:06:46 +0000123
Francis Beaudetd92b9471999-01-30 13:24:23 +0000124 return newStream;
125}
126
127/***
128 * This is the destructor of the StgStreamImpl class.
129 *
Vincent Béron9a624912002-05-31 23:06:46 +0000130 * This method will clean-up all the resources used-up by the given StgStreamImpl
Francis Beaudetd92b9471999-01-30 13:24:23 +0000131 * class. The pointer passed-in to this function will be freed and will not
132 * be valid anymore.
133 */
134void StgStreamImpl_Destroy(StgStreamImpl* This)
135{
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000136 TRACE("(%p)\n", This);
Francis Beaudetec2edc71999-05-02 11:41:10 +0000137
Francis Beaudetd92b9471999-01-30 13:24:23 +0000138 /*
139 * Release the reference we are holding on the parent storage.
140 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000141 IStorage_Release((IStorage*)This->parentStorage);
Francis Beaudetd92b9471999-01-30 13:24:23 +0000142 This->parentStorage = 0;
143
144 /*
145 * Make sure we clean-up the block chain stream objects that we were using.
146 */
147 if (This->bigBlockChain != 0)
148 {
149 BlockChainStream_Destroy(This->bigBlockChain);
150 This->bigBlockChain = 0;
151 }
152
153 if (This->smallBlockChain != 0)
154 {
155 SmallBlockChainStream_Destroy(This->smallBlockChain);
156 This->smallBlockChain = 0;
157 }
158
159 /*
160 * Finally, free the memory used-up by the class.
161 */
Vincent Béron9a624912002-05-31 23:06:46 +0000162 HeapFree(GetProcessHeap(), 0, This);
Francis Beaudetd92b9471999-01-30 13:24:23 +0000163}
164
165/***
166 * This implements the IUnknown method QueryInterface for this
167 * class
168 */
169HRESULT WINAPI StgStreamImpl_QueryInterface(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000170 IStream* iface,
Vincent Béron9a624912002-05-31 23:06:46 +0000171 REFIID riid, /* [in] */
172 void** ppvObject) /* [iid_is][out] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000173{
Francois Gougetb0c61291999-02-18 13:26:22 +0000174 StgStreamImpl* const This=(StgStreamImpl*)iface;
175
Francis Beaudetd92b9471999-01-30 13:24:23 +0000176 /*
177 * Perform a sanity check on the parameters.
178 */
179 if (ppvObject==0)
180 return E_INVALIDARG;
Vincent Béron9a624912002-05-31 23:06:46 +0000181
Francis Beaudetd92b9471999-01-30 13:24:23 +0000182 /*
183 * Initialize the return parameter.
184 */
185 *ppvObject = 0;
Vincent Béron9a624912002-05-31 23:06:46 +0000186
Francis Beaudetd92b9471999-01-30 13:24:23 +0000187 /*
188 * Compare the riid with the interface IDs implemented by this object.
189 */
Vincent Béron9a624912002-05-31 23:06:46 +0000190 if (memcmp(&IID_IUnknown, riid, sizeof(IID_IUnknown)) == 0)
Francis Beaudetd92b9471999-01-30 13:24:23 +0000191 {
Alexandre Julliarda3960291999-02-26 11:11:13 +0000192 *ppvObject = (IStream*)This;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000193 }
Vincent Béron9a624912002-05-31 23:06:46 +0000194 else if (memcmp(&IID_IStream, riid, sizeof(IID_IStream)) == 0)
Francis Beaudetd92b9471999-01-30 13:24:23 +0000195 {
Alexandre Julliarda3960291999-02-26 11:11:13 +0000196 *ppvObject = (IStream*)This;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000197 }
Vincent Béron9a624912002-05-31 23:06:46 +0000198
Francis Beaudetd92b9471999-01-30 13:24:23 +0000199 /*
200 * Check that we obtained an interface.
201 */
202 if ((*ppvObject)==0)
203 return E_NOINTERFACE;
Vincent Béron9a624912002-05-31 23:06:46 +0000204
Francis Beaudetd92b9471999-01-30 13:24:23 +0000205 /*
206 * Query Interface always increases the reference count by one when it is
207 * successful
208 */
Francois Gougetb0c61291999-02-18 13:26:22 +0000209 StgStreamImpl_AddRef(iface);
Vincent Béron9a624912002-05-31 23:06:46 +0000210
211 return S_OK;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000212}
213
214/***
215 * This implements the IUnknown method AddRef for this
216 * class
217 */
218ULONG WINAPI StgStreamImpl_AddRef(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000219 IStream* iface)
Francis Beaudetd92b9471999-01-30 13:24:23 +0000220{
Francois Gougetb0c61291999-02-18 13:26:22 +0000221 StgStreamImpl* const This=(StgStreamImpl*)iface;
222
Francis Beaudetd92b9471999-01-30 13:24:23 +0000223 This->ref++;
Vincent Béron9a624912002-05-31 23:06:46 +0000224
Francis Beaudetd92b9471999-01-30 13:24:23 +0000225 return This->ref;
226}
227
228/***
229 * This implements the IUnknown method Release for this
230 * class
231 */
232ULONG WINAPI StgStreamImpl_Release(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000233 IStream* iface)
Francis Beaudetd92b9471999-01-30 13:24:23 +0000234{
Francois Gougetb0c61291999-02-18 13:26:22 +0000235 StgStreamImpl* const This=(StgStreamImpl*)iface;
236
Francis Beaudetd92b9471999-01-30 13:24:23 +0000237 ULONG newRef;
Vincent Béron9a624912002-05-31 23:06:46 +0000238
Francis Beaudetd92b9471999-01-30 13:24:23 +0000239 This->ref--;
Vincent Béron9a624912002-05-31 23:06:46 +0000240
Francis Beaudetd92b9471999-01-30 13:24:23 +0000241 newRef = This->ref;
Vincent Béron9a624912002-05-31 23:06:46 +0000242
Francis Beaudetd92b9471999-01-30 13:24:23 +0000243 /*
244 * If the reference count goes down to 0, perform suicide.
245 */
246 if (newRef==0)
247 {
248 StgStreamImpl_Destroy(This);
249 }
Vincent Béron9a624912002-05-31 23:06:46 +0000250
Francis Beaudetd92b9471999-01-30 13:24:23 +0000251 return newRef;
252}
253
254/***
255 * This method will open the block chain pointed by the property
256 * that describes the stream.
257 * If the stream's size is null, no chain is opened.
258 */
259void StgStreamImpl_OpenBlockChain(
260 StgStreamImpl* This)
261{
262 StgProperty curProperty;
Alexandre Julliarda3960291999-02-26 11:11:13 +0000263 BOOL readSucessful;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000264
265 /*
Alexandre Julliard925d6022002-01-31 21:01:35 +0000266 * Make sure no old object is left over.
Francis Beaudetd92b9471999-01-30 13:24:23 +0000267 */
268 if (This->smallBlockChain != 0)
269 {
270 SmallBlockChainStream_Destroy(This->smallBlockChain);
271 This->smallBlockChain = 0;
272 }
273
274 if (This->bigBlockChain != 0)
275 {
276 BlockChainStream_Destroy(This->bigBlockChain);
277 This->bigBlockChain = 0;
278 }
279
280 /*
281 * Read the information from the property.
282 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000283 readSucessful = StorageImpl_ReadProperty(This->parentStorage->ancestorStorage,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000284 This->ownerProperty,
285 &curProperty);
Vincent Béron9a624912002-05-31 23:06:46 +0000286
Francis Beaudetd92b9471999-01-30 13:24:23 +0000287 if (readSucessful)
288 {
289 This->streamSize = curProperty.size;
Vincent Béron9a624912002-05-31 23:06:46 +0000290
Francis Beaudetd92b9471999-01-30 13:24:23 +0000291 /*
292 * This code supports only streams that are <32 bits in size.
293 */
Patrik Stridvall311e4561999-09-19 14:20:33 +0000294 assert(This->streamSize.s.HighPart == 0);
Vincent Béron9a624912002-05-31 23:06:46 +0000295
Francis Beaudetd92b9471999-01-30 13:24:23 +0000296 if(curProperty.startingBlock == BLOCK_END_OF_CHAIN)
297 {
Patrik Stridvall311e4561999-09-19 14:20:33 +0000298 assert( (This->streamSize.s.HighPart == 0) && (This->streamSize.s.LowPart == 0) );
Francis Beaudetd92b9471999-01-30 13:24:23 +0000299 }
300 else
301 {
Patrik Stridvall311e4561999-09-19 14:20:33 +0000302 if ( (This->streamSize.s.HighPart == 0) &&
303 (This->streamSize.s.LowPart < LIMIT_TO_USE_SMALL_BLOCK) )
Francis Beaudetd92b9471999-01-30 13:24:23 +0000304 {
305 This->smallBlockChain = SmallBlockChainStream_Construct(
Vincent Béron9a624912002-05-31 23:06:46 +0000306 This->parentStorage->ancestorStorage,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000307 This->ownerProperty);
Francis Beaudetd92b9471999-01-30 13:24:23 +0000308 }
309 else
310 {
311 This->bigBlockChain = BlockChainStream_Construct(
312 This->parentStorage->ancestorStorage,
313 NULL,
314 This->ownerProperty);
315 }
316 }
317 }
318}
319
320/***
321 * This method is part of the ISequentialStream interface.
322 *
Alexandre Julliard925d6022002-01-31 21:01:35 +0000323 * It reads a block of information from the stream at the current
Francis Beaudetd92b9471999-01-30 13:24:23 +0000324 * position. It then moves the current position at the end of the
325 * read block
326 *
327 * See the documentation of ISequentialStream for more info.
328 */
Vincent Béron9a624912002-05-31 23:06:46 +0000329HRESULT WINAPI StgStreamImpl_Read(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000330 IStream* iface,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000331 void* pv, /* [length_is][size_is][out] */
Vincent Béron9a624912002-05-31 23:06:46 +0000332 ULONG cb, /* [in] */
333 ULONG* pcbRead) /* [out] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000334{
Francois Gougetb0c61291999-02-18 13:26:22 +0000335 StgStreamImpl* const This=(StgStreamImpl*)iface;
336
Francis Beaudetd92b9471999-01-30 13:24:23 +0000337 ULONG bytesReadBuffer;
338 ULONG bytesToReadFromBuffer;
Alexandre Julliard925d6022002-01-31 21:01:35 +0000339 HRESULT res = S_FALSE;
Francis Beaudetec2edc71999-05-02 11:41:10 +0000340
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000341 TRACE("(%p, %p, %ld, %p)\n",
Francis Beaudetec2edc71999-05-02 11:41:10 +0000342 iface, pv, cb, pcbRead);
343
Vincent Béron9a624912002-05-31 23:06:46 +0000344 /*
Alexandre Julliard925d6022002-01-31 21:01:35 +0000345 * If the caller is not interested in the number of bytes read,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000346 * we use another buffer to avoid "if" statements in the code.
347 */
348 if (pcbRead==0)
349 pcbRead = &bytesReadBuffer;
Vincent Béron9a624912002-05-31 23:06:46 +0000350
Francis Beaudetd92b9471999-01-30 13:24:23 +0000351 /*
352 * Using the known size of the stream, calculate the number of bytes
353 * to read from the block chain
354 */
Francois Gouget6d77d3a2000-03-25 21:44:35 +0000355 bytesToReadFromBuffer = min( This->streamSize.s.LowPart - This->currentPosition.s.LowPart, cb);
Vincent Béron9a624912002-05-31 23:06:46 +0000356
Francis Beaudetd92b9471999-01-30 13:24:23 +0000357 /*
358 * Depending on the type of chain that was opened when the stream was constructed,
Alexandre Julliard925d6022002-01-31 21:01:35 +0000359 * we delegate the work to the method that reads the block chains.
Francis Beaudetd92b9471999-01-30 13:24:23 +0000360 */
361 if (This->smallBlockChain!=0)
362 {
363 SmallBlockChainStream_ReadAt(This->smallBlockChain,
364 This->currentPosition,
365 bytesToReadFromBuffer,
366 pv,
367 pcbRead);
Vincent Béron9a624912002-05-31 23:06:46 +0000368
Francis Beaudetd92b9471999-01-30 13:24:23 +0000369 }
370 else if (This->bigBlockChain!=0)
371 {
372 BlockChainStream_ReadAt(This->bigBlockChain,
373 This->currentPosition,
374 bytesToReadFromBuffer,
375 pv,
376 pcbRead);
377 }
378 else
Thuy Nguyen71b327f1999-07-10 10:13:50 +0000379 {
380 /*
381 * Small and big block chains are both NULL. This case will happen
382 * when a stream starts with BLOCK_END_OF_CHAIN and has size zero.
383 */
384
385 *pcbRead = 0;
Alexandre Julliard925d6022002-01-31 21:01:35 +0000386 res = S_OK;
387 goto end;
Thuy Nguyen71b327f1999-07-10 10:13:50 +0000388 }
Francis Beaudetd92b9471999-01-30 13:24:23 +0000389
390 /*
391 * We should always be able to read the proper amount of data from the
392 * chain.
393 */
394 assert(bytesToReadFromBuffer == *pcbRead);
395
396 /*
397 * Advance the pointer for the number of positions read.
398 */
Patrik Stridvall311e4561999-09-19 14:20:33 +0000399 This->currentPosition.s.LowPart += *pcbRead;
Vincent Béron9a624912002-05-31 23:06:46 +0000400
Alexandre Julliard925d6022002-01-31 21:01:35 +0000401 if(*pcbRead != cb)
402 {
403 WARN("read %ld instead of the required %ld bytes !\n", *pcbRead, cb);
404 /*
405 * this used to return S_FALSE, however MSDN docu says that an app should
406 * be prepared to handle error in case of stream end reached, as *some*
407 * implementations *might* return an error (IOW: most do *not*).
408 * As some program fails on returning S_FALSE, I better use S_OK here.
409 */
410 res = S_OK;
411 }
412 else
413 res = S_OK;
Vincent Béron9a624912002-05-31 23:06:46 +0000414
Alexandre Julliard925d6022002-01-31 21:01:35 +0000415end:
416 TRACE("<-- %08lx\n", res);
417 return res;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000418}
Vincent Béron9a624912002-05-31 23:06:46 +0000419
Francis Beaudetd92b9471999-01-30 13:24:23 +0000420/***
421 * This method is part of the ISequentialStream interface.
422 *
423 * It writes a block of information to the stream at the current
424 * position. It then moves the current position at the end of the
425 * written block. If the stream is too small to fit the block,
426 * the stream is grown to fit.
427 *
428 * See the documentation of ISequentialStream for more info.
429 */
430HRESULT WINAPI StgStreamImpl_Write(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000431 IStream* iface,
Vincent Béron9a624912002-05-31 23:06:46 +0000432 const void* pv, /* [size_is][in] */
433 ULONG cb, /* [in] */
434 ULONG* pcbWritten) /* [out] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000435{
Francois Gougetb0c61291999-02-18 13:26:22 +0000436 StgStreamImpl* const This=(StgStreamImpl*)iface;
437
Francis Beaudetd92b9471999-01-30 13:24:23 +0000438 ULARGE_INTEGER newSize;
439 ULONG bytesWritten = 0;
Francis Beaudetec2edc71999-05-02 11:41:10 +0000440
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000441 TRACE("(%p, %p, %ld, %p)\n",
Francis Beaudetec2edc71999-05-02 11:41:10 +0000442 iface, pv, cb, pcbWritten);
Vincent Béron9a624912002-05-31 23:06:46 +0000443
Huw D M Daviesbdee2122000-05-03 17:39:21 +0000444 /*
445 * Do we have permission to write to this stream?
446 */
447 if (!(This->grfMode & (STGM_WRITE | STGM_READWRITE))) {
448 return STG_E_ACCESSDENIED;
449 }
Gerard Patel617f6902000-01-26 02:04:44 +0000450
Francis Beaudetd92b9471999-01-30 13:24:23 +0000451 /*
452 * If the caller is not interested in the number of bytes written,
453 * we use another buffer to avoid "if" statements in the code.
454 */
455 if (pcbWritten == 0)
456 pcbWritten = &bytesWritten;
Vincent Béron9a624912002-05-31 23:06:46 +0000457
Francis Beaudetec2edc71999-05-02 11:41:10 +0000458 /*
459 * Initialize the out parameter
460 */
461 *pcbWritten = 0;
462
Francis Beaudetd92b9471999-01-30 13:24:23 +0000463 if (cb == 0)
464 {
465 return S_OK;
466 }
467 else
468 {
Patrik Stridvall311e4561999-09-19 14:20:33 +0000469 newSize.s.HighPart = 0;
470 newSize.s.LowPart = This->currentPosition.s.LowPart + cb;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000471 }
Vincent Béron9a624912002-05-31 23:06:46 +0000472
Francis Beaudetd92b9471999-01-30 13:24:23 +0000473 /*
474 * Verify if we need to grow the stream
475 */
Patrik Stridvall311e4561999-09-19 14:20:33 +0000476 if (newSize.s.LowPart > This->streamSize.s.LowPart)
Francis Beaudetd92b9471999-01-30 13:24:23 +0000477 {
478 /* grow stream */
Francis Beaudetec2edc71999-05-02 11:41:10 +0000479 IStream_SetSize(iface, newSize);
Francis Beaudetd92b9471999-01-30 13:24:23 +0000480 }
Vincent Béron9a624912002-05-31 23:06:46 +0000481
Francis Beaudetd92b9471999-01-30 13:24:23 +0000482 /*
483 * Depending on the type of chain that was opened when the stream was constructed,
484 * we delegate the work to the method that readwrites to the block chains.
485 */
486 if (This->smallBlockChain!=0)
487 {
488 SmallBlockChainStream_WriteAt(This->smallBlockChain,
489 This->currentPosition,
490 cb,
491 pv,
492 pcbWritten);
Vincent Béron9a624912002-05-31 23:06:46 +0000493
Francis Beaudetd92b9471999-01-30 13:24:23 +0000494 }
495 else if (This->bigBlockChain!=0)
496 {
497 BlockChainStream_WriteAt(This->bigBlockChain,
498 This->currentPosition,
499 cb,
500 pv,
501 pcbWritten);
502 }
503 else
504 assert(FALSE);
Vincent Béron9a624912002-05-31 23:06:46 +0000505
Francis Beaudetd92b9471999-01-30 13:24:23 +0000506 /*
507 * Advance the position pointer for the number of positions written.
508 */
Patrik Stridvall311e4561999-09-19 14:20:33 +0000509 This->currentPosition.s.LowPart += *pcbWritten;
Vincent Béron9a624912002-05-31 23:06:46 +0000510
Francis Beaudetd92b9471999-01-30 13:24:23 +0000511 return S_OK;
512}
513
514/***
515 * This method is part of the IStream interface.
516 *
517 * It will move the current stream pointer according to the parameters
518 * given.
519 *
520 * See the documentation of IStream for more info.
Vincent Béron9a624912002-05-31 23:06:46 +0000521 */
522HRESULT WINAPI StgStreamImpl_Seek(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000523 IStream* iface,
Vincent Béron9a624912002-05-31 23:06:46 +0000524 LARGE_INTEGER dlibMove, /* [in] */
525 DWORD dwOrigin, /* [in] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000526 ULARGE_INTEGER* plibNewPosition) /* [out] */
527{
Francois Gougetb0c61291999-02-18 13:26:22 +0000528 StgStreamImpl* const This=(StgStreamImpl*)iface;
529
Francis Beaudetd92b9471999-01-30 13:24:23 +0000530 ULARGE_INTEGER newPosition;
531
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000532 TRACE("(%p, %ld, %ld, %p)\n",
Patrik Stridvall311e4561999-09-19 14:20:33 +0000533 iface, dlibMove.s.LowPart, dwOrigin, plibNewPosition);
Francis Beaudetec2edc71999-05-02 11:41:10 +0000534
Vincent Béron9a624912002-05-31 23:06:46 +0000535 /*
Francis Beaudetd92b9471999-01-30 13:24:23 +0000536 * The caller is allowed to pass in NULL as the new position return value.
537 * If it happens, we assign it to a dynamic variable to avoid special cases
538 * in the code below.
539 */
540 if (plibNewPosition == 0)
541 {
542 plibNewPosition = &newPosition;
543 }
544
545 /*
546 * The file pointer is moved depending on the given "function"
547 * parameter.
548 */
549 switch (dwOrigin)
550 {
551 case STREAM_SEEK_SET:
Patrik Stridvall311e4561999-09-19 14:20:33 +0000552 plibNewPosition->s.HighPart = 0;
553 plibNewPosition->s.LowPart = 0;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000554 break;
555 case STREAM_SEEK_CUR:
556 *plibNewPosition = This->currentPosition;
557 break;
558 case STREAM_SEEK_END:
559 *plibNewPosition = This->streamSize;
560 break;
561 default:
562 return STG_E_INVALIDFUNCTION;
563 }
564
Alexandre Julliard27952ef2000-10-13 20:26:03 +0000565 plibNewPosition->QuadPart = RtlLargeIntegerAdd( plibNewPosition->QuadPart, dlibMove.QuadPart );
Pierre Mageau89aa8612000-02-25 20:58:25 +0000566
Francis Beaudetd92b9471999-01-30 13:24:23 +0000567 /*
Pierre Mageau89aa8612000-02-25 20:58:25 +0000568 * tell the caller what we calculated
Francis Beaudetd92b9471999-01-30 13:24:23 +0000569 */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000570 This->currentPosition = *plibNewPosition;
Vincent Béron9a624912002-05-31 23:06:46 +0000571
Francis Beaudetd92b9471999-01-30 13:24:23 +0000572 return S_OK;
573}
574
575/***
576 * This method is part of the IStream interface.
577 *
578 * It will change the size of a stream.
579 *
580 * TODO: Switch from small blocks to big blocks and vice versa.
581 *
582 * See the documentation of IStream for more info.
583 */
Vincent Béron9a624912002-05-31 23:06:46 +0000584HRESULT WINAPI StgStreamImpl_SetSize(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000585 IStream* iface,
Vincent Béron9a624912002-05-31 23:06:46 +0000586 ULARGE_INTEGER libNewSize) /* [in] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000587{
Francois Gougetb0c61291999-02-18 13:26:22 +0000588 StgStreamImpl* const This=(StgStreamImpl*)iface;
589
Francis Beaudetd92b9471999-01-30 13:24:23 +0000590 StgProperty curProperty;
Alexandre Julliarda3960291999-02-26 11:11:13 +0000591 BOOL Success;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000592
Patrik Stridvall311e4561999-09-19 14:20:33 +0000593 TRACE("(%p, %ld)\n", iface, libNewSize.s.LowPart);
Francis Beaudetec2edc71999-05-02 11:41:10 +0000594
Francis Beaudetd92b9471999-01-30 13:24:23 +0000595 /*
596 * As documented.
597 */
Patrik Stridvall311e4561999-09-19 14:20:33 +0000598 if (libNewSize.s.HighPart != 0)
Francis Beaudetd92b9471999-01-30 13:24:23 +0000599 return STG_E_INVALIDFUNCTION;
Pierre Mageau89aa8612000-02-25 20:58:25 +0000600
601 /*
602 * Do we have permission?
603 */
604 if (!(This->grfMode & (STGM_WRITE | STGM_READWRITE)))
605 return STG_E_ACCESSDENIED;
606
Patrik Stridvall311e4561999-09-19 14:20:33 +0000607 if (This->streamSize.s.LowPart == libNewSize.s.LowPart)
Francis Beaudetd92b9471999-01-30 13:24:23 +0000608 return S_OK;
609
610 /*
611 * This will happen if we're creating a stream
612 */
613 if ((This->smallBlockChain == 0) && (This->bigBlockChain == 0))
614 {
Patrik Stridvall311e4561999-09-19 14:20:33 +0000615 if (libNewSize.s.LowPart < LIMIT_TO_USE_SMALL_BLOCK)
Francis Beaudetd92b9471999-01-30 13:24:23 +0000616 {
617 This->smallBlockChain = SmallBlockChainStream_Construct(
618 This->parentStorage->ancestorStorage,
619 This->ownerProperty);
620 }
621 else
622 {
623 This->bigBlockChain = BlockChainStream_Construct(
624 This->parentStorage->ancestorStorage,
625 NULL,
626 This->ownerProperty);
627 }
628 }
629
630 /*
631 * Read this stream's property to see if it's small blocks or big blocks
632 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000633 Success = StorageImpl_ReadProperty(This->parentStorage->ancestorStorage,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000634 This->ownerProperty,
Vincent Béron9a624912002-05-31 23:06:46 +0000635 &curProperty);
Francis Beaudetd92b9471999-01-30 13:24:23 +0000636 /*
Thuy Nguyen251c9df1999-02-13 12:10:15 +0000637 * Determine if we have to switch from small to big blocks or vice versa
Vincent Béron9a624912002-05-31 23:06:46 +0000638 */
639 if ( (This->smallBlockChain!=0) &&
Patrik Stridvall311e4561999-09-19 14:20:33 +0000640 (curProperty.size.s.LowPart < LIMIT_TO_USE_SMALL_BLOCK) )
Thuy Nguyen251c9df1999-02-13 12:10:15 +0000641 {
Patrik Stridvall311e4561999-09-19 14:20:33 +0000642 if (libNewSize.s.LowPart >= LIMIT_TO_USE_SMALL_BLOCK)
Thuy Nguyen251c9df1999-02-13 12:10:15 +0000643 {
644 /*
645 * Transform the small block chain into a big block chain
646 */
647 This->bigBlockChain = Storage32Impl_SmallBlocksToBigBlocks(
648 This->parentStorage->ancestorStorage,
649 &This->smallBlockChain);
650 }
651 }
652
Francis Beaudetd92b9471999-01-30 13:24:23 +0000653 if (This->smallBlockChain!=0)
654 {
655 Success = SmallBlockChainStream_SetSize(This->smallBlockChain, libNewSize);
Francis Beaudetd92b9471999-01-30 13:24:23 +0000656 }
657 else
658 {
659 Success = BlockChainStream_SetSize(This->bigBlockChain, libNewSize);
Francis Beaudetd92b9471999-01-30 13:24:23 +0000660 }
661
662 /*
Alexandre Julliard925d6022002-01-31 21:01:35 +0000663 * Write the new information about this stream to the property
Francis Beaudetd92b9471999-01-30 13:24:23 +0000664 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000665 Success = StorageImpl_ReadProperty(This->parentStorage->ancestorStorage,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000666 This->ownerProperty,
667 &curProperty);
668
Patrik Stridvall311e4561999-09-19 14:20:33 +0000669 curProperty.size.s.HighPart = libNewSize.s.HighPart;
670 curProperty.size.s.LowPart = libNewSize.s.LowPart;
Vincent Béron9a624912002-05-31 23:06:46 +0000671
Francis Beaudetd92b9471999-01-30 13:24:23 +0000672 if (Success)
673 {
Alexandre Julliarda3960291999-02-26 11:11:13 +0000674 StorageImpl_WriteProperty(This->parentStorage->ancestorStorage,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000675 This->ownerProperty,
676 &curProperty);
677 }
Vincent Béron9a624912002-05-31 23:06:46 +0000678
Francis Beaudetd92b9471999-01-30 13:24:23 +0000679 This->streamSize = libNewSize;
Vincent Béron9a624912002-05-31 23:06:46 +0000680
Francis Beaudetd92b9471999-01-30 13:24:23 +0000681 return S_OK;
682}
Vincent Béron9a624912002-05-31 23:06:46 +0000683
Stephane Lussier07e242f1999-04-15 15:47:48 +0000684/***
685 * This method is part of the IStream interface.
686 *
687 * It will copy the 'cb' Bytes to 'pstm' IStream.
688 *
689 * See the documentation of IStream for more info.
690 */
Vincent Béron9a624912002-05-31 23:06:46 +0000691HRESULT WINAPI StgStreamImpl_CopyTo(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000692 IStream* iface,
Vincent Béron9a624912002-05-31 23:06:46 +0000693 IStream* pstm, /* [unique][in] */
694 ULARGE_INTEGER cb, /* [in] */
695 ULARGE_INTEGER* pcbRead, /* [out] */
696 ULARGE_INTEGER* pcbWritten) /* [out] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000697{
Stephane Lussier07e242f1999-04-15 15:47:48 +0000698 HRESULT hr = S_OK;
699 BYTE tmpBuffer[128];
700 ULONG bytesRead, bytesWritten, copySize;
701 ULARGE_INTEGER totalBytesRead;
702 ULARGE_INTEGER totalBytesWritten;
703
Vincent Béron9a624912002-05-31 23:06:46 +0000704 TRACE("(%p, %p, %ld, %p, %p)\n",
Patrik Stridvall311e4561999-09-19 14:20:33 +0000705 iface, pstm, cb.s.LowPart, pcbRead, pcbWritten);
Francis Beaudetec2edc71999-05-02 11:41:10 +0000706
Stephane Lussier07e242f1999-04-15 15:47:48 +0000707 /*
708 * Sanity check
709 */
710 if ( pstm == 0 )
711 return STG_E_INVALIDPOINTER;
712
Patrik Stridvall311e4561999-09-19 14:20:33 +0000713 totalBytesRead.s.LowPart = totalBytesRead.s.HighPart = 0;
714 totalBytesWritten.s.LowPart = totalBytesWritten.s.HighPart = 0;
Stephane Lussier07e242f1999-04-15 15:47:48 +0000715
716 /*
Alexandre Julliard925d6022002-01-31 21:01:35 +0000717 * use stack to store data temporarily
718 * there is surely a more performant way of doing it, for now this basic
Stephane Lussier07e242f1999-04-15 15:47:48 +0000719 * implementation will do the job
720 */
Patrik Stridvall311e4561999-09-19 14:20:33 +0000721 while ( cb.s.LowPart > 0 )
Stephane Lussier07e242f1999-04-15 15:47:48 +0000722 {
Patrik Stridvall311e4561999-09-19 14:20:33 +0000723 if ( cb.s.LowPart >= 128 )
Stephane Lussier07e242f1999-04-15 15:47:48 +0000724 copySize = 128;
725 else
Patrik Stridvall311e4561999-09-19 14:20:33 +0000726 copySize = cb.s.LowPart;
Vincent Béron9a624912002-05-31 23:06:46 +0000727
Francis Beaudetec2edc71999-05-02 11:41:10 +0000728 IStream_Read(iface, tmpBuffer, copySize, &bytesRead);
Stephane Lussier07e242f1999-04-15 15:47:48 +0000729
Patrik Stridvall311e4561999-09-19 14:20:33 +0000730 totalBytesRead.s.LowPart += bytesRead;
Vincent Béron9a624912002-05-31 23:06:46 +0000731
Francis Beaudetec2edc71999-05-02 11:41:10 +0000732 IStream_Write(pstm, tmpBuffer, bytesRead, &bytesWritten);
Stephane Lussier07e242f1999-04-15 15:47:48 +0000733
Patrik Stridvall311e4561999-09-19 14:20:33 +0000734 totalBytesWritten.s.LowPart += bytesWritten;
Stephane Lussier07e242f1999-04-15 15:47:48 +0000735
736 /*
Alexandre Julliard925d6022002-01-31 21:01:35 +0000737 * Check that read & write operations were successful
Stephane Lussier07e242f1999-04-15 15:47:48 +0000738 */
Francis Beaudetec2edc71999-05-02 11:41:10 +0000739 if (bytesRead != bytesWritten)
Stephane Lussier07e242f1999-04-15 15:47:48 +0000740 {
741 hr = STG_E_MEDIUMFULL;
742 break;
743 }
Vincent Béron9a624912002-05-31 23:06:46 +0000744
Francis Beaudetec2edc71999-05-02 11:41:10 +0000745 if (bytesRead!=copySize)
Patrik Stridvall311e4561999-09-19 14:20:33 +0000746 cb.s.LowPart = 0;
Francis Beaudetec2edc71999-05-02 11:41:10 +0000747 else
Patrik Stridvall311e4561999-09-19 14:20:33 +0000748 cb.s.LowPart -= bytesRead;
Stephane Lussier07e242f1999-04-15 15:47:48 +0000749 }
750
751 /*
752 * Update number of bytes read and written
753 */
754 if (pcbRead)
755 {
Patrik Stridvall311e4561999-09-19 14:20:33 +0000756 pcbRead->s.LowPart = totalBytesRead.s.LowPart;
757 pcbRead->s.HighPart = totalBytesRead.s.HighPart;
Stephane Lussier07e242f1999-04-15 15:47:48 +0000758 }
759
760 if (pcbWritten)
761 {
Patrik Stridvall311e4561999-09-19 14:20:33 +0000762 pcbWritten->s.LowPart = totalBytesWritten.s.LowPart;
763 pcbWritten->s.HighPart = totalBytesWritten.s.HighPart;
Stephane Lussier07e242f1999-04-15 15:47:48 +0000764 }
765 return hr;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000766}
767
768/***
769 * This method is part of the IStream interface.
770 *
771 * For streams contained in structured storages, this method
772 * does nothing. This is what the documentation tells us.
773 *
774 * See the documentation of IStream for more info.
Vincent Béron9a624912002-05-31 23:06:46 +0000775 */
776HRESULT WINAPI StgStreamImpl_Commit(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000777 IStream* iface,
Vincent Béron9a624912002-05-31 23:06:46 +0000778 DWORD grfCommitFlags) /* [in] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000779{
780 return S_OK;
781}
782
783/***
784 * This method is part of the IStream interface.
785 *
786 * For streams contained in structured storages, this method
787 * does nothing. This is what the documentation tells us.
788 *
789 * See the documentation of IStream for more info.
Vincent Béron9a624912002-05-31 23:06:46 +0000790 */
791HRESULT WINAPI StgStreamImpl_Revert(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000792 IStream* iface)
Francis Beaudetd92b9471999-01-30 13:24:23 +0000793{
794 return S_OK;
795}
796
Vincent Béron9a624912002-05-31 23:06:46 +0000797HRESULT WINAPI StgStreamImpl_LockRegion(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000798 IStream* iface,
Vincent Béron9a624912002-05-31 23:06:46 +0000799 ULARGE_INTEGER libOffset, /* [in] */
800 ULARGE_INTEGER cb, /* [in] */
801 DWORD dwLockType) /* [in] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000802{
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000803 FIXME("not implemented!\n");
Francis Beaudetd92b9471999-01-30 13:24:23 +0000804 return E_NOTIMPL;
805}
806
Vincent Béron9a624912002-05-31 23:06:46 +0000807HRESULT WINAPI StgStreamImpl_UnlockRegion(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000808 IStream* iface,
Vincent Béron9a624912002-05-31 23:06:46 +0000809 ULARGE_INTEGER libOffset, /* [in] */
810 ULARGE_INTEGER cb, /* [in] */
811 DWORD dwLockType) /* [in] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000812{
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000813 FIXME("not implemented!\n");
Francis Beaudetd92b9471999-01-30 13:24:23 +0000814 return E_NOTIMPL;
815}
816
817/***
818 * This method is part of the IStream interface.
819 *
820 * This method returns information about the current
821 * stream.
822 *
823 * See the documentation of IStream for more info.
Vincent Béron9a624912002-05-31 23:06:46 +0000824 */
825HRESULT WINAPI StgStreamImpl_Stat(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000826 IStream* iface,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000827 STATSTG* pstatstg, /* [out] */
Vincent Béron9a624912002-05-31 23:06:46 +0000828 DWORD grfStatFlag) /* [in] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000829{
Francois Gougetb0c61291999-02-18 13:26:22 +0000830 StgStreamImpl* const This=(StgStreamImpl*)iface;
831
Francis Beaudetd92b9471999-01-30 13:24:23 +0000832 StgProperty curProperty;
Alexandre Julliarda3960291999-02-26 11:11:13 +0000833 BOOL readSucessful;
Vincent Béron9a624912002-05-31 23:06:46 +0000834
Francis Beaudetd92b9471999-01-30 13:24:23 +0000835 /*
836 * Read the information from the property.
837 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000838 readSucessful = StorageImpl_ReadProperty(This->parentStorage->ancestorStorage,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000839 This->ownerProperty,
840 &curProperty);
Vincent Béron9a624912002-05-31 23:06:46 +0000841
Francis Beaudetd92b9471999-01-30 13:24:23 +0000842 if (readSucessful)
843 {
Vincent Béron9a624912002-05-31 23:06:46 +0000844 StorageUtl_CopyPropertyToSTATSTG(pstatstg,
845 &curProperty,
Francis Beaudetd92b9471999-01-30 13:24:23 +0000846 grfStatFlag);
Pierre Mageau89aa8612000-02-25 20:58:25 +0000847
848 pstatstg->grfMode = This->grfMode;
Vincent Béron9a624912002-05-31 23:06:46 +0000849
Francis Beaudetd92b9471999-01-30 13:24:23 +0000850 return S_OK;
851 }
Vincent Béron9a624912002-05-31 23:06:46 +0000852
Francis Beaudetd92b9471999-01-30 13:24:23 +0000853 return E_FAIL;
854}
Vincent Béron9a624912002-05-31 23:06:46 +0000855
Bill Medland548d8a22001-09-10 23:14:39 +0000856/***
857 * This method is part of the IStream interface.
858 *
859 * This method returns a clone of the interface that allows for
860 * another seek pointer
861 *
862 * See the documentation of IStream for more info.
863 *
864 * I am not totally sure what I am doing here but I presume that this
865 * should be basically as simple as creating a new stream with the same
866 * parent etc and positioning its seek cursor.
Vincent Béron9a624912002-05-31 23:06:46 +0000867 */
868HRESULT WINAPI StgStreamImpl_Clone(
Alexandre Julliarda3960291999-02-26 11:11:13 +0000869 IStream* iface,
Vincent Béron9a624912002-05-31 23:06:46 +0000870 IStream** ppstm) /* [out] */
Francis Beaudetd92b9471999-01-30 13:24:23 +0000871{
Bill Medland548d8a22001-09-10 23:14:39 +0000872 StgStreamImpl* const This=(StgStreamImpl*)iface;
873 HRESULT hres;
874 StgStreamImpl* new_stream;
875 LARGE_INTEGER seek_pos;
876
877 /*
878 * Sanity check
879 */
880 if ( ppstm == 0 )
881 return STG_E_INVALIDPOINTER;
882
883 new_stream = StgStreamImpl_Construct (This->parentStorage, This->grfMode, This->ownerProperty);
884
885 if (!new_stream)
886 return STG_E_INSUFFICIENTMEMORY; /* Currently the only reason for new_stream=0 */
887
888 *ppstm = (IStream*) new_stream;
889 seek_pos.QuadPart = This->currentPosition.QuadPart;
Vincent Béron9a624912002-05-31 23:06:46 +0000890
Bill Medland548d8a22001-09-10 23:14:39 +0000891 hres=StgStreamImpl_Seek (*ppstm, seek_pos, STREAM_SEEK_SET, NULL);
Vincent Béron9a624912002-05-31 23:06:46 +0000892
Bill Medland548d8a22001-09-10 23:14:39 +0000893 assert (SUCCEEDED(hres));
894
895 return S_OK;
Francis Beaudetd92b9471999-01-30 13:24:23 +0000896}