Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 1 | /* |
| 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 Julliard | 0799c1a | 2002-03-09 23:29:33 +0000 | [diff] [blame] | 10 | * |
| 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 Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 24 | */ |
Dimitrie O. Paun | 297f3d8 | 2003-01-07 20:36:20 +0000 | [diff] [blame] | 25 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 26 | #include <assert.h> |
| 27 | #include <stdlib.h> |
| 28 | #include <stdio.h> |
| 29 | #include <string.h> |
| 30 | |
Dimitrie O. Paun | 297f3d8 | 2003-01-07 20:36:20 +0000 | [diff] [blame] | 31 | #define NONAMELESSUNION |
| 32 | #define NONAMELESSSTRUCT |
Marcus Meissner | 064f170 | 1999-02-28 19:14:33 +0000 | [diff] [blame] | 33 | #include "winbase.h" |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 34 | #include "winerror.h" |
Patrik Stridvall | 9c1de6d | 2002-09-12 22:07:02 +0000 | [diff] [blame] | 35 | #include "winternl.h" |
Alexandre Julliard | 0799c1a | 2002-03-09 23:29:33 +0000 | [diff] [blame] | 36 | #include "wine/debug.h" |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 37 | |
| 38 | #include "storage32.h" |
| 39 | |
Alexandre Julliard | 0799c1a | 2002-03-09 23:29:33 +0000 | [diff] [blame] | 40 | WINE_DEFAULT_DEBUG_CHANNEL(storage); |
Patrik Stridvall | b4b9fae | 1999-04-19 14:56:29 +0000 | [diff] [blame] | 41 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 42 | |
| 43 | /* |
| 44 | * Virtual function table for the StgStreamImpl class. |
| 45 | */ |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 46 | static ICOM_VTABLE(IStream) StgStreamImpl_Vtbl = |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 47 | { |
Paul Quinn | 2305f3c | 1999-05-22 11:41:38 +0000 | [diff] [blame] | 48 | ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 49 | 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 Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 63 | }; |
| 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 | */ |
| 76 | StgStreamImpl* StgStreamImpl_Construct( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 77 | StorageBaseImpl* parentStorage, |
Pierre Mageau | 89aa861 | 2000-02-25 20:58:25 +0000 | [diff] [blame] | 78 | DWORD grfMode, |
| 79 | ULONG ownerProperty) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 80 | { |
| 81 | StgStreamImpl* newStream; |
| 82 | |
| 83 | newStream = HeapAlloc(GetProcessHeap(), 0, sizeof(StgStreamImpl)); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 84 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 85 | if (newStream!=0) |
| 86 | { |
| 87 | /* |
| 88 | * Set-up the virtual function table and reference count. |
| 89 | */ |
Francois Gouget | 01c9ac4 | 1999-10-31 01:59:23 +0000 | [diff] [blame] | 90 | ICOM_VTBL(newStream) = &StgStreamImpl_Vtbl; |
| 91 | newStream->ref = 0; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 92 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 93 | /* |
| 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 Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 98 | IStorage_AddRef((IStorage*)newStream->parentStorage); |
Pierre Mageau | 89aa861 | 2000-02-25 20:58:25 +0000 | [diff] [blame] | 99 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 100 | newStream->grfMode = grfMode; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 101 | newStream->ownerProperty = ownerProperty; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 102 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 103 | /* |
Francois Gouget | 3bb9a36 | 2001-10-22 19:04:32 +0000 | [diff] [blame] | 104 | * Start the stream at the beginning. |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 105 | */ |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 106 | newStream->currentPosition.s.HighPart = 0; |
| 107 | newStream->currentPosition.s.LowPart = 0; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 108 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 109 | /* |
| 110 | * Initialize the rest of the data. |
| 111 | */ |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 112 | newStream->streamSize.s.HighPart = 0; |
| 113 | newStream->streamSize.s.LowPart = 0; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 114 | newStream->bigBlockChain = 0; |
| 115 | newStream->smallBlockChain = 0; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 116 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 117 | /* |
| 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 123 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 124 | return newStream; |
| 125 | } |
| 126 | |
| 127 | /*** |
| 128 | * This is the destructor of the StgStreamImpl class. |
| 129 | * |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 130 | * This method will clean-up all the resources used-up by the given StgStreamImpl |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 131 | * class. The pointer passed-in to this function will be freed and will not |
| 132 | * be valid anymore. |
| 133 | */ |
| 134 | void StgStreamImpl_Destroy(StgStreamImpl* This) |
| 135 | { |
Alexandre Julliard | 359f497e | 1999-07-04 16:02:24 +0000 | [diff] [blame] | 136 | TRACE("(%p)\n", This); |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 137 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 138 | /* |
| 139 | * Release the reference we are holding on the parent storage. |
| 140 | */ |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 141 | IStorage_Release((IStorage*)This->parentStorage); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 142 | 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 162 | HeapFree(GetProcessHeap(), 0, This); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 163 | } |
| 164 | |
| 165 | /*** |
| 166 | * This implements the IUnknown method QueryInterface for this |
| 167 | * class |
| 168 | */ |
| 169 | HRESULT WINAPI StgStreamImpl_QueryInterface( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 170 | IStream* iface, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 171 | REFIID riid, /* [in] */ |
| 172 | void** ppvObject) /* [iid_is][out] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 173 | { |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 174 | StgStreamImpl* const This=(StgStreamImpl*)iface; |
| 175 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 176 | /* |
| 177 | * Perform a sanity check on the parameters. |
| 178 | */ |
| 179 | if (ppvObject==0) |
| 180 | return E_INVALIDARG; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 181 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 182 | /* |
| 183 | * Initialize the return parameter. |
| 184 | */ |
| 185 | *ppvObject = 0; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 186 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 187 | /* |
| 188 | * Compare the riid with the interface IDs implemented by this object. |
| 189 | */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 190 | if (memcmp(&IID_IUnknown, riid, sizeof(IID_IUnknown)) == 0) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 191 | { |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 192 | *ppvObject = (IStream*)This; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 193 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 194 | else if (memcmp(&IID_IStream, riid, sizeof(IID_IStream)) == 0) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 195 | { |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 196 | *ppvObject = (IStream*)This; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 197 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 198 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 199 | /* |
| 200 | * Check that we obtained an interface. |
| 201 | */ |
| 202 | if ((*ppvObject)==0) |
| 203 | return E_NOINTERFACE; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 204 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 205 | /* |
| 206 | * Query Interface always increases the reference count by one when it is |
| 207 | * successful |
| 208 | */ |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 209 | StgStreamImpl_AddRef(iface); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 210 | |
| 211 | return S_OK; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 212 | } |
| 213 | |
| 214 | /*** |
| 215 | * This implements the IUnknown method AddRef for this |
| 216 | * class |
| 217 | */ |
| 218 | ULONG WINAPI StgStreamImpl_AddRef( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 219 | IStream* iface) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 220 | { |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 221 | StgStreamImpl* const This=(StgStreamImpl*)iface; |
| 222 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 223 | This->ref++; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 224 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 225 | return This->ref; |
| 226 | } |
| 227 | |
| 228 | /*** |
| 229 | * This implements the IUnknown method Release for this |
| 230 | * class |
| 231 | */ |
| 232 | ULONG WINAPI StgStreamImpl_Release( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 233 | IStream* iface) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 234 | { |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 235 | StgStreamImpl* const This=(StgStreamImpl*)iface; |
| 236 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 237 | ULONG newRef; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 238 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 239 | This->ref--; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 240 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 241 | newRef = This->ref; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 242 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 243 | /* |
| 244 | * If the reference count goes down to 0, perform suicide. |
| 245 | */ |
| 246 | if (newRef==0) |
| 247 | { |
| 248 | StgStreamImpl_Destroy(This); |
| 249 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 250 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 251 | 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 | */ |
| 259 | void StgStreamImpl_OpenBlockChain( |
| 260 | StgStreamImpl* This) |
| 261 | { |
| 262 | StgProperty curProperty; |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 263 | BOOL readSucessful; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 264 | |
| 265 | /* |
Alexandre Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 266 | * Make sure no old object is left over. |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 267 | */ |
| 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 Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 283 | readSucessful = StorageImpl_ReadProperty(This->parentStorage->ancestorStorage, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 284 | This->ownerProperty, |
| 285 | &curProperty); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 286 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 287 | if (readSucessful) |
| 288 | { |
| 289 | This->streamSize = curProperty.size; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 290 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 291 | /* |
| 292 | * This code supports only streams that are <32 bits in size. |
| 293 | */ |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 294 | assert(This->streamSize.s.HighPart == 0); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 295 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 296 | if(curProperty.startingBlock == BLOCK_END_OF_CHAIN) |
| 297 | { |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 298 | assert( (This->streamSize.s.HighPart == 0) && (This->streamSize.s.LowPart == 0) ); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 299 | } |
| 300 | else |
| 301 | { |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 302 | if ( (This->streamSize.s.HighPart == 0) && |
| 303 | (This->streamSize.s.LowPart < LIMIT_TO_USE_SMALL_BLOCK) ) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 304 | { |
| 305 | This->smallBlockChain = SmallBlockChainStream_Construct( |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 306 | This->parentStorage->ancestorStorage, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 307 | This->ownerProperty); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 308 | } |
| 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 Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 323 | * It reads a block of information from the stream at the current |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 324 | * 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 329 | HRESULT WINAPI StgStreamImpl_Read( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 330 | IStream* iface, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 331 | void* pv, /* [length_is][size_is][out] */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 332 | ULONG cb, /* [in] */ |
| 333 | ULONG* pcbRead) /* [out] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 334 | { |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 335 | StgStreamImpl* const This=(StgStreamImpl*)iface; |
| 336 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 337 | ULONG bytesReadBuffer; |
| 338 | ULONG bytesToReadFromBuffer; |
Alexandre Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 339 | HRESULT res = S_FALSE; |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 340 | |
Alexandre Julliard | 359f497e | 1999-07-04 16:02:24 +0000 | [diff] [blame] | 341 | TRACE("(%p, %p, %ld, %p)\n", |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 342 | iface, pv, cb, pcbRead); |
| 343 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 344 | /* |
Alexandre Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 345 | * If the caller is not interested in the number of bytes read, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 346 | * we use another buffer to avoid "if" statements in the code. |
| 347 | */ |
| 348 | if (pcbRead==0) |
| 349 | pcbRead = &bytesReadBuffer; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 350 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 351 | /* |
| 352 | * Using the known size of the stream, calculate the number of bytes |
| 353 | * to read from the block chain |
| 354 | */ |
Francois Gouget | 6d77d3a | 2000-03-25 21:44:35 +0000 | [diff] [blame] | 355 | bytesToReadFromBuffer = min( This->streamSize.s.LowPart - This->currentPosition.s.LowPart, cb); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 356 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 357 | /* |
| 358 | * Depending on the type of chain that was opened when the stream was constructed, |
Alexandre Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 359 | * we delegate the work to the method that reads the block chains. |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 360 | */ |
| 361 | if (This->smallBlockChain!=0) |
| 362 | { |
| 363 | SmallBlockChainStream_ReadAt(This->smallBlockChain, |
| 364 | This->currentPosition, |
| 365 | bytesToReadFromBuffer, |
| 366 | pv, |
| 367 | pcbRead); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 368 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 369 | } |
| 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 Nguyen | 71b327f | 1999-07-10 10:13:50 +0000 | [diff] [blame] | 379 | { |
| 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 Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 386 | res = S_OK; |
| 387 | goto end; |
Thuy Nguyen | 71b327f | 1999-07-10 10:13:50 +0000 | [diff] [blame] | 388 | } |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 389 | |
| 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 Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 399 | This->currentPosition.s.LowPart += *pcbRead; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 400 | |
Alexandre Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 401 | 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 414 | |
Alexandre Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 415 | end: |
| 416 | TRACE("<-- %08lx\n", res); |
| 417 | return res; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 418 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 419 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 420 | /*** |
| 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 | */ |
| 430 | HRESULT WINAPI StgStreamImpl_Write( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 431 | IStream* iface, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 432 | const void* pv, /* [size_is][in] */ |
| 433 | ULONG cb, /* [in] */ |
| 434 | ULONG* pcbWritten) /* [out] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 435 | { |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 436 | StgStreamImpl* const This=(StgStreamImpl*)iface; |
| 437 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 438 | ULARGE_INTEGER newSize; |
| 439 | ULONG bytesWritten = 0; |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 440 | |
Alexandre Julliard | 359f497e | 1999-07-04 16:02:24 +0000 | [diff] [blame] | 441 | TRACE("(%p, %p, %ld, %p)\n", |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 442 | iface, pv, cb, pcbWritten); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 443 | |
Huw D M Davies | bdee212 | 2000-05-03 17:39:21 +0000 | [diff] [blame] | 444 | /* |
| 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 Patel | 617f690 | 2000-01-26 02:04:44 +0000 | [diff] [blame] | 450 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 451 | /* |
| 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 457 | |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 458 | /* |
| 459 | * Initialize the out parameter |
| 460 | */ |
| 461 | *pcbWritten = 0; |
| 462 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 463 | if (cb == 0) |
| 464 | { |
| 465 | return S_OK; |
| 466 | } |
| 467 | else |
| 468 | { |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 469 | newSize.s.HighPart = 0; |
| 470 | newSize.s.LowPart = This->currentPosition.s.LowPart + cb; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 471 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 472 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 473 | /* |
| 474 | * Verify if we need to grow the stream |
| 475 | */ |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 476 | if (newSize.s.LowPart > This->streamSize.s.LowPart) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 477 | { |
| 478 | /* grow stream */ |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 479 | IStream_SetSize(iface, newSize); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 480 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 481 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 482 | /* |
| 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 493 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 494 | } |
| 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 505 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 506 | /* |
| 507 | * Advance the position pointer for the number of positions written. |
| 508 | */ |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 509 | This->currentPosition.s.LowPart += *pcbWritten; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 510 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 511 | 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 521 | */ |
| 522 | HRESULT WINAPI StgStreamImpl_Seek( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 523 | IStream* iface, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 524 | LARGE_INTEGER dlibMove, /* [in] */ |
| 525 | DWORD dwOrigin, /* [in] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 526 | ULARGE_INTEGER* plibNewPosition) /* [out] */ |
| 527 | { |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 528 | StgStreamImpl* const This=(StgStreamImpl*)iface; |
| 529 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 530 | ULARGE_INTEGER newPosition; |
| 531 | |
Alexandre Julliard | 359f497e | 1999-07-04 16:02:24 +0000 | [diff] [blame] | 532 | TRACE("(%p, %ld, %ld, %p)\n", |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 533 | iface, dlibMove.s.LowPart, dwOrigin, plibNewPosition); |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 534 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 535 | /* |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 536 | * 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 Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 552 | plibNewPosition->s.HighPart = 0; |
| 553 | plibNewPosition->s.LowPart = 0; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 554 | 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 Julliard | 27952ef | 2000-10-13 20:26:03 +0000 | [diff] [blame] | 565 | plibNewPosition->QuadPart = RtlLargeIntegerAdd( plibNewPosition->QuadPart, dlibMove.QuadPart ); |
Pierre Mageau | 89aa861 | 2000-02-25 20:58:25 +0000 | [diff] [blame] | 566 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 567 | /* |
Pierre Mageau | 89aa861 | 2000-02-25 20:58:25 +0000 | [diff] [blame] | 568 | * tell the caller what we calculated |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 569 | */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 570 | This->currentPosition = *plibNewPosition; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 571 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 572 | 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 584 | HRESULT WINAPI StgStreamImpl_SetSize( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 585 | IStream* iface, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 586 | ULARGE_INTEGER libNewSize) /* [in] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 587 | { |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 588 | StgStreamImpl* const This=(StgStreamImpl*)iface; |
| 589 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 590 | StgProperty curProperty; |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 591 | BOOL Success; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 592 | |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 593 | TRACE("(%p, %ld)\n", iface, libNewSize.s.LowPart); |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 594 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 595 | /* |
| 596 | * As documented. |
| 597 | */ |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 598 | if (libNewSize.s.HighPart != 0) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 599 | return STG_E_INVALIDFUNCTION; |
Pierre Mageau | 89aa861 | 2000-02-25 20:58:25 +0000 | [diff] [blame] | 600 | |
| 601 | /* |
| 602 | * Do we have permission? |
| 603 | */ |
| 604 | if (!(This->grfMode & (STGM_WRITE | STGM_READWRITE))) |
| 605 | return STG_E_ACCESSDENIED; |
| 606 | |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 607 | if (This->streamSize.s.LowPart == libNewSize.s.LowPart) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 608 | 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 Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 615 | if (libNewSize.s.LowPart < LIMIT_TO_USE_SMALL_BLOCK) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 616 | { |
| 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 Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 633 | Success = StorageImpl_ReadProperty(This->parentStorage->ancestorStorage, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 634 | This->ownerProperty, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 635 | &curProperty); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 636 | /* |
Thuy Nguyen | 251c9df | 1999-02-13 12:10:15 +0000 | [diff] [blame] | 637 | * Determine if we have to switch from small to big blocks or vice versa |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 638 | */ |
| 639 | if ( (This->smallBlockChain!=0) && |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 640 | (curProperty.size.s.LowPart < LIMIT_TO_USE_SMALL_BLOCK) ) |
Thuy Nguyen | 251c9df | 1999-02-13 12:10:15 +0000 | [diff] [blame] | 641 | { |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 642 | if (libNewSize.s.LowPart >= LIMIT_TO_USE_SMALL_BLOCK) |
Thuy Nguyen | 251c9df | 1999-02-13 12:10:15 +0000 | [diff] [blame] | 643 | { |
| 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 Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 653 | if (This->smallBlockChain!=0) |
| 654 | { |
| 655 | Success = SmallBlockChainStream_SetSize(This->smallBlockChain, libNewSize); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 656 | } |
| 657 | else |
| 658 | { |
| 659 | Success = BlockChainStream_SetSize(This->bigBlockChain, libNewSize); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 660 | } |
| 661 | |
| 662 | /* |
Alexandre Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 663 | * Write the new information about this stream to the property |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 664 | */ |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 665 | Success = StorageImpl_ReadProperty(This->parentStorage->ancestorStorage, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 666 | This->ownerProperty, |
| 667 | &curProperty); |
| 668 | |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 669 | curProperty.size.s.HighPart = libNewSize.s.HighPart; |
| 670 | curProperty.size.s.LowPart = libNewSize.s.LowPart; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 671 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 672 | if (Success) |
| 673 | { |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 674 | StorageImpl_WriteProperty(This->parentStorage->ancestorStorage, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 675 | This->ownerProperty, |
| 676 | &curProperty); |
| 677 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 678 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 679 | This->streamSize = libNewSize; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 680 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 681 | return S_OK; |
| 682 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 683 | |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 684 | /*** |
| 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 691 | HRESULT WINAPI StgStreamImpl_CopyTo( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 692 | IStream* iface, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 693 | IStream* pstm, /* [unique][in] */ |
| 694 | ULARGE_INTEGER cb, /* [in] */ |
| 695 | ULARGE_INTEGER* pcbRead, /* [out] */ |
| 696 | ULARGE_INTEGER* pcbWritten) /* [out] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 697 | { |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 698 | HRESULT hr = S_OK; |
| 699 | BYTE tmpBuffer[128]; |
| 700 | ULONG bytesRead, bytesWritten, copySize; |
| 701 | ULARGE_INTEGER totalBytesRead; |
| 702 | ULARGE_INTEGER totalBytesWritten; |
| 703 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 704 | TRACE("(%p, %p, %ld, %p, %p)\n", |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 705 | iface, pstm, cb.s.LowPart, pcbRead, pcbWritten); |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 706 | |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 707 | /* |
| 708 | * Sanity check |
| 709 | */ |
| 710 | if ( pstm == 0 ) |
| 711 | return STG_E_INVALIDPOINTER; |
| 712 | |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 713 | totalBytesRead.s.LowPart = totalBytesRead.s.HighPart = 0; |
| 714 | totalBytesWritten.s.LowPart = totalBytesWritten.s.HighPart = 0; |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 715 | |
| 716 | /* |
Alexandre Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 717 | * use stack to store data temporarily |
| 718 | * there is surely a more performant way of doing it, for now this basic |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 719 | * implementation will do the job |
| 720 | */ |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 721 | while ( cb.s.LowPart > 0 ) |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 722 | { |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 723 | if ( cb.s.LowPart >= 128 ) |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 724 | copySize = 128; |
| 725 | else |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 726 | copySize = cb.s.LowPart; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 727 | |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 728 | IStream_Read(iface, tmpBuffer, copySize, &bytesRead); |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 729 | |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 730 | totalBytesRead.s.LowPart += bytesRead; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 731 | |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 732 | IStream_Write(pstm, tmpBuffer, bytesRead, &bytesWritten); |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 733 | |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 734 | totalBytesWritten.s.LowPart += bytesWritten; |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 735 | |
| 736 | /* |
Alexandre Julliard | 925d602 | 2002-01-31 21:01:35 +0000 | [diff] [blame] | 737 | * Check that read & write operations were successful |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 738 | */ |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 739 | if (bytesRead != bytesWritten) |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 740 | { |
| 741 | hr = STG_E_MEDIUMFULL; |
| 742 | break; |
| 743 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 744 | |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 745 | if (bytesRead!=copySize) |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 746 | cb.s.LowPart = 0; |
Francis Beaudet | ec2edc7 | 1999-05-02 11:41:10 +0000 | [diff] [blame] | 747 | else |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 748 | cb.s.LowPart -= bytesRead; |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 749 | } |
| 750 | |
| 751 | /* |
| 752 | * Update number of bytes read and written |
| 753 | */ |
| 754 | if (pcbRead) |
| 755 | { |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 756 | pcbRead->s.LowPart = totalBytesRead.s.LowPart; |
| 757 | pcbRead->s.HighPart = totalBytesRead.s.HighPart; |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 758 | } |
| 759 | |
| 760 | if (pcbWritten) |
| 761 | { |
Patrik Stridvall | 311e456 | 1999-09-19 14:20:33 +0000 | [diff] [blame] | 762 | pcbWritten->s.LowPart = totalBytesWritten.s.LowPart; |
| 763 | pcbWritten->s.HighPart = totalBytesWritten.s.HighPart; |
Stephane Lussier | 07e242f | 1999-04-15 15:47:48 +0000 | [diff] [blame] | 764 | } |
| 765 | return hr; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 766 | } |
| 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 775 | */ |
| 776 | HRESULT WINAPI StgStreamImpl_Commit( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 777 | IStream* iface, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 778 | DWORD grfCommitFlags) /* [in] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 779 | { |
| 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 790 | */ |
| 791 | HRESULT WINAPI StgStreamImpl_Revert( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 792 | IStream* iface) |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 793 | { |
| 794 | return S_OK; |
| 795 | } |
| 796 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 797 | HRESULT WINAPI StgStreamImpl_LockRegion( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 798 | IStream* iface, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 799 | ULARGE_INTEGER libOffset, /* [in] */ |
| 800 | ULARGE_INTEGER cb, /* [in] */ |
| 801 | DWORD dwLockType) /* [in] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 802 | { |
Alexandre Julliard | 359f497e | 1999-07-04 16:02:24 +0000 | [diff] [blame] | 803 | FIXME("not implemented!\n"); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 804 | return E_NOTIMPL; |
| 805 | } |
| 806 | |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 807 | HRESULT WINAPI StgStreamImpl_UnlockRegion( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 808 | IStream* iface, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 809 | ULARGE_INTEGER libOffset, /* [in] */ |
| 810 | ULARGE_INTEGER cb, /* [in] */ |
| 811 | DWORD dwLockType) /* [in] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 812 | { |
Alexandre Julliard | 359f497e | 1999-07-04 16:02:24 +0000 | [diff] [blame] | 813 | FIXME("not implemented!\n"); |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 814 | 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 824 | */ |
| 825 | HRESULT WINAPI StgStreamImpl_Stat( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 826 | IStream* iface, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 827 | STATSTG* pstatstg, /* [out] */ |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 828 | DWORD grfStatFlag) /* [in] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 829 | { |
Francois Gouget | b0c6129 | 1999-02-18 13:26:22 +0000 | [diff] [blame] | 830 | StgStreamImpl* const This=(StgStreamImpl*)iface; |
| 831 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 832 | StgProperty curProperty; |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 833 | BOOL readSucessful; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 834 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 835 | /* |
| 836 | * Read the information from the property. |
| 837 | */ |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 838 | readSucessful = StorageImpl_ReadProperty(This->parentStorage->ancestorStorage, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 839 | This->ownerProperty, |
| 840 | &curProperty); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 841 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 842 | if (readSucessful) |
| 843 | { |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 844 | StorageUtl_CopyPropertyToSTATSTG(pstatstg, |
| 845 | &curProperty, |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 846 | grfStatFlag); |
Pierre Mageau | 89aa861 | 2000-02-25 20:58:25 +0000 | [diff] [blame] | 847 | |
| 848 | pstatstg->grfMode = This->grfMode; |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 849 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 850 | return S_OK; |
| 851 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 852 | |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 853 | return E_FAIL; |
| 854 | } |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 855 | |
Bill Medland | 548d8a2 | 2001-09-10 23:14:39 +0000 | [diff] [blame] | 856 | /*** |
| 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 867 | */ |
| 868 | HRESULT WINAPI StgStreamImpl_Clone( |
Alexandre Julliard | a396029 | 1999-02-26 11:11:13 +0000 | [diff] [blame] | 869 | IStream* iface, |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 870 | IStream** ppstm) /* [out] */ |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 871 | { |
Bill Medland | 548d8a2 | 2001-09-10 23:14:39 +0000 | [diff] [blame] | 872 | 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éron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 890 | |
Bill Medland | 548d8a2 | 2001-09-10 23:14:39 +0000 | [diff] [blame] | 891 | hres=StgStreamImpl_Seek (*ppstm, seek_pos, STREAM_SEEK_SET, NULL); |
Vincent Béron | 9a62491 | 2002-05-31 23:06:46 +0000 | [diff] [blame] | 892 | |
Bill Medland | 548d8a2 | 2001-09-10 23:14:39 +0000 | [diff] [blame] | 893 | assert (SUCCEEDED(hres)); |
| 894 | |
| 895 | return S_OK; |
Francis Beaudet | d92b947 | 1999-01-30 13:24:23 +0000 | [diff] [blame] | 896 | } |