1 /*****************************************************************************/ 2 // Copyright 2006-2007 Adobe Systems Incorporated 3 // All Rights Reserved. 4 // 5 // NOTICE: Adobe permits you to use, modify, and distribute this file in 6 // accordance with the terms of the Adobe license agreement accompanying it. 7 /*****************************************************************************/ 8 9 /* $Id: //mondo/dng_sdk_1_4/dng_sdk/source/dng_memory_stream.cpp#1 $ */ 10 /* $DateTime: 2012/05/30 13:28:51 $ */ 11 /* $Change: 832332 $ */ 12 /* $Author: tknoll $ */ 13 14 /*****************************************************************************/ 15 16 #include "dng_memory_stream.h" 17 18 #include "dng_bottlenecks.h" 19 #include "dng_exceptions.h" 20 #include "dng_safe_arithmetic.h" 21 #include "dng_utils.h" 22 23 /*****************************************************************************/ 24 25 dng_memory_stream::dng_memory_stream (dng_memory_allocator &allocator, 26 dng_abort_sniffer *sniffer, 27 uint32 pageSize) 28 29 : dng_stream (sniffer, 30 kDefaultBufferSize, 31 kDNGStreamInvalidOffset) 32 33 , fAllocator (allocator) 34 , fPageSize (pageSize ) 35 36 , fPageCount (0) 37 , fPagesAllocated (0) 38 , fPageList (NULL) 39 40 , fMemoryStreamLength (0) 41 42 { 43 44 } 45 46 /*****************************************************************************/ 47 48 dng_memory_stream::~dng_memory_stream () 49 { 50 51 if (fPageList) 52 { 53 54 for (uint32 index = 0; index < fPageCount; index++) 55 { 56 57 delete fPageList [index]; 58 59 } 60 61 free (fPageList); 62 63 } 64 65 } 66 67 /*****************************************************************************/ 68 69 uint64 dng_memory_stream::DoGetLength () 70 { 71 72 return fMemoryStreamLength; 73 74 } 75 76 /*****************************************************************************/ 77 78 void dng_memory_stream::DoRead (void *data, 79 uint32 count, 80 uint64 offset) 81 { 82 83 if (offset + count > fMemoryStreamLength) 84 { 85 86 ThrowEndOfFile (); 87 88 } 89 90 uint64 baseOffset = offset; 91 92 while (count) 93 { 94 95 uint32 pageIndex = (uint32) (offset / fPageSize); 96 uint32 pageOffset = (uint32) (offset % fPageSize); 97 98 uint32 blockCount = Min_uint32 (fPageSize - pageOffset, count); 99 100 const uint8 *sPtr = fPageList [pageIndex]->Buffer_uint8 () + 101 pageOffset; 102 103 uint8 *dPtr = ((uint8 *) data) + (uint32) (offset - baseOffset); 104 105 DoCopyBytes (sPtr, dPtr, blockCount); 106 107 offset += blockCount; 108 count -= blockCount; 109 110 } 111 112 } 113 114 /*****************************************************************************/ 115 116 void dng_memory_stream::DoSetLength (uint64 length) 117 { 118 119 while (length > fPageCount * (uint64) fPageSize) 120 { 121 122 if (fPageCount == fPagesAllocated) 123 { 124 125 uint32 newSizeTemp1 = 0, newSizeTemp2 = 0; 126 if (!SafeUint32Add (fPagesAllocated, 32u, &newSizeTemp1) || 127 !SafeUint32Mult (fPagesAllocated, 2u, &newSizeTemp2)) 128 { 129 ThrowMemoryFull ("Arithmetic overflow in DoSetLength()"); 130 } 131 uint32 newSize = Max_uint32 (newSizeTemp1, newSizeTemp2); 132 uint32 numBytes; 133 if (!SafeUint32Mult (newSize, sizeof (dng_memory_block *), 134 &numBytes)) 135 { 136 ThrowMemoryFull ("Arithmetic overflow in DoSetLength()"); 137 } 138 139 dng_memory_block **list = (dng_memory_block **) malloc (numBytes); 140 141 if (!list) 142 { 143 144 ThrowMemoryFull (); 145 146 } 147 148 if (fPageCount) 149 { 150 151 // The multiplication here is safe against overflow. fPageCount 152 // can never reach a value that is large enough to cause 153 // overflow because the computation of numBytes above would fail 154 // before a list of that size could be allocated. 155 DoCopyBytes (fPageList, 156 list, 157 fPageCount * (uint32) sizeof (dng_memory_block *)); 158 159 } 160 161 if (fPageList) 162 { 163 164 free (fPageList); 165 166 } 167 168 fPageList = list; 169 170 fPagesAllocated = newSize; 171 172 } 173 174 fPageList [fPageCount] = fAllocator.Allocate (fPageSize); 175 176 fPageCount++; 177 178 } 179 180 fMemoryStreamLength = length; 181 182 } 183 184 /*****************************************************************************/ 185 186 void dng_memory_stream::DoWrite (const void *data, 187 uint32 count, 188 uint64 offset) 189 { 190 191 DoSetLength (Max_uint64 (fMemoryStreamLength, 192 offset + count)); 193 194 uint64 baseOffset = offset; 195 196 while (count) 197 { 198 199 uint32 pageIndex = (uint32) (offset / fPageSize); 200 uint32 pageOffset = (uint32) (offset % fPageSize); 201 202 uint32 blockCount = Min_uint32 (fPageSize - pageOffset, count); 203 204 const uint8 *sPtr = ((const uint8 *) data) + (uint32) (offset - baseOffset); 205 206 uint8 *dPtr = fPageList [pageIndex]->Buffer_uint8 () + 207 pageOffset; 208 209 DoCopyBytes (sPtr, dPtr, blockCount); 210 211 offset += blockCount; 212 count -= blockCount; 213 214 } 215 216 } 217 218 /*****************************************************************************/ 219 220 void dng_memory_stream::CopyToStream (dng_stream &dstStream, 221 uint64 count) 222 { 223 224 if (count < kBigBufferSize) 225 { 226 227 dng_stream::CopyToStream (dstStream, count); 228 229 } 230 231 else 232 { 233 234 Flush (); 235 236 uint64 offset = Position (); 237 238 if (offset + count > Length ()) 239 { 240 241 ThrowEndOfFile (); 242 243 } 244 245 while (count) 246 { 247 248 uint32 pageIndex = (uint32) (offset / fPageSize); 249 uint32 pageOffset = (uint32) (offset % fPageSize); 250 251 uint32 blockCount = (uint32) Min_uint64 (fPageSize - pageOffset, count); 252 253 const uint8 *sPtr = fPageList [pageIndex]->Buffer_uint8 () + 254 pageOffset; 255 256 dstStream.Put (sPtr, blockCount); 257 258 offset += blockCount; 259 count -= blockCount; 260 261 } 262 263 SetReadPosition (offset); 264 265 } 266 267 } 268 269 /*****************************************************************************/ 270