1 /*
2  * Copyright 2009 Tony Wasserka
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2.1 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17  */
18 
19 #include "wine/test.h"
20 
21 #define COBJMACROS
22 #include "wincodec.h"
23 
24 static void test_StreamOnMemory(void)
25 {
26     IWICImagingFactory *pFactory;
27     IWICStream *pStream, *pBufStream;
28     const BYTE CmpMem[] = {
29         0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
30         0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
31         0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
32         0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
33     };
34     const BYTE CmpMemOverlap[] = {
35         0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
36         0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
37         0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
38         0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
39     };
40     const BYTE ZeroMem[10] = {0};
41     BYTE Memory[64], MemBuf[64];
42     LARGE_INTEGER LargeNull, LargeInt, SeekPos;
43     ULARGE_INTEGER uLargeNull, uNewPos;
44     ULONG uBytesRead, uBytesWritten;
45     HRESULT hr;
46     STATSTG Stats;
47 
48     LargeNull.QuadPart = 0;
49     uLargeNull.QuadPart = 0;
50     SeekPos.QuadPart = 5;
51 
52     memcpy(Memory, CmpMem, sizeof(CmpMem));
53 
54     CoInitialize(NULL);
55 
56     hr = CoCreateInstance(&CLSID_WICImagingFactory, NULL, CLSCTX_INPROC_SERVER, &IID_IWICImagingFactory, (void**)&pFactory);
57     if(FAILED(hr)) {
58         skip("CoCreateInstance returned with %#x, expected %#x\n", hr, S_OK);
59         return;
60     }
61 
62     hr = IWICImagingFactory_CreateStream(pFactory, &pStream);
63     ok(hr == S_OK, "CreateStream returned with %#x, expected %#x\n", hr, S_OK);
64     if(FAILED(hr)) {
65         skip("Failed to create stream\n");
66         return;
67     }
68 
69     /* InitializeFromMemory */
70     hr = IWICStream_InitializeFromMemory(pStream, NULL, sizeof(Memory));   /* memory = NULL */
71     ok(hr == E_INVALIDARG, "InitializeFromMemory returned with %#x, expected %#x\n", hr, E_INVALIDARG);
72 
73     hr = IWICStream_InitializeFromMemory(pStream, Memory, 0);   /* size = 0 */
74     ok(hr == S_OK, "InitializeFromMemory returned with %#x, expected %#x\n", hr, S_OK);
75 
76     hr = IWICStream_InitializeFromMemory(pStream, Memory, sizeof(Memory));   /* stream already initialized */
77     ok(hr == WINCODEC_ERR_WRONGSTATE, "InitializeFromMemory returned with %#x, expected %#x\n", hr, WINCODEC_ERR_WRONGSTATE);
78 
79     /* recreate stream */
80     IWICStream_Release(pStream);
81     hr = IWICImagingFactory_CreateStream(pFactory, &pStream);
82     ok(hr == S_OK, "CreateStream failed with %#x\n", hr);
83 
84     hr = IWICStream_InitializeFromMemory(pStream, Memory, sizeof(Memory));
85     ok(hr == S_OK, "InitializeFromMemory returned with %#x, expected %#x\n", hr, S_OK);
86 
87     /* IWICStream does not maintain an independent copy of the backing memory buffer. */
88     memcpy(Memory, ZeroMem, sizeof(ZeroMem));
89     hr = IWICStream_Read(pStream, MemBuf, sizeof(ZeroMem), &uBytesRead);
90     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
91     if(SUCCEEDED(hr)) {
92         ok(uBytesRead == sizeof(ZeroMem), "Read %u bytes\n", uBytesRead);
93         ok(memcmp(MemBuf, ZeroMem, sizeof(ZeroMem)) == 0, "Read returned invalid data!\n");
94     }
95 
96     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
97 
98     hr = IWICStream_Write(pStream, CmpMem, sizeof(CmpMem), &uBytesWritten);
99     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
100 
101     /* Seek */
102     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, &uNewPos);
103     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
104     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
105 
106     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
107     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
108 
109     LargeInt.u.HighPart = 1;
110     LargeInt.u.LowPart = 0;
111     uNewPos.u.HighPart = 0xdeadbeef;
112     uNewPos.u.LowPart = 0xdeadbeef;
113     hr = IWICStream_Seek(pStream, LargeInt, STREAM_SEEK_SET, &uNewPos);
114     ok(hr == HRESULT_FROM_WIN32(ERROR_ARITHMETIC_OVERFLOW), "Seek returned with %#x, expected %#x\n", hr, HRESULT_FROM_WIN32(ERROR_ARITHMETIC_OVERFLOW));
115     ok(uNewPos.u.HighPart == 0xdeadbeef && uNewPos.u.LowPart == 0xdeadbeef, "Seek cursor initialized to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0xdeadbeef, 0xdeadbeef);
116     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
117     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
118     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
119 
120     LargeInt.QuadPart = sizeof(Memory) + 10;
121     uNewPos.u.HighPart = 0xdeadbeef;
122     uNewPos.u.LowPart = 0xdeadbeef;
123     hr = IWICStream_Seek(pStream, LargeInt, STREAM_SEEK_SET, &uNewPos);
124     ok(hr == E_INVALIDARG, "Seek returned with %#x, expected %#x\n", hr, E_INVALIDARG);
125     ok(uNewPos.u.HighPart == 0xdeadbeef && uNewPos.u.LowPart == 0xdeadbeef, "Seek cursor initialized to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0xdeadbeef, 0xdeadbeef);
126     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
127     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
128     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
129 
130     LargeInt.QuadPart = 1;
131     uNewPos.u.HighPart = 0xdeadbeef;
132     uNewPos.u.LowPart = 0xdeadbeef;
133     hr = IWICStream_Seek(pStream, LargeInt, STREAM_SEEK_END, &uNewPos);
134     ok(hr == E_INVALIDARG, "Seek returned with %#x, expected %#x\n", hr, E_INVALIDARG);
135     ok(uNewPos.u.HighPart == 0xdeadbeef && uNewPos.u.LowPart == 0xdeadbeef, "Seek cursor initialized to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0xdeadbeef, 0xdeadbeef);
136     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
137     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
138     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
139 
140     LargeInt.QuadPart = -1;
141     hr = IWICStream_Seek(pStream, LargeInt, STREAM_SEEK_END, &uNewPos);
142     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
143     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == sizeof(Memory) - 1, "bSeek cursor moved to position (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart);
144 
145     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, &uNewPos); /* reset seek pointer */
146     LargeInt.QuadPart = -(LONGLONG)sizeof(Memory) - 5;
147     uNewPos.u.HighPart = 0xdeadbeef;
148     uNewPos.u.LowPart = 0xdeadbeef;
149     hr = IWICStream_Seek(pStream, LargeInt, STREAM_SEEK_END, &uNewPos);
150     ok(hr == HRESULT_FROM_WIN32(ERROR_ARITHMETIC_OVERFLOW),
151        "Seek returned with %#x, expected %#x\n", hr, HRESULT_FROM_WIN32(ERROR_ARITHMETIC_OVERFLOW));
152     ok(uNewPos.u.HighPart == 0xdeadbeef && uNewPos.u.LowPart == 0xdeadbeef, "Seek cursor initialized to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0xdeadbeef, 0xdeadbeef);
153     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
154     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
155     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0); /* remains unchanged */
156     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
157 
158 
159     /* Read */
160     hr = IWICStream_Read(pStream, MemBuf, 12, &uBytesRead);
161     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
162     if(SUCCEEDED(hr)) {
163         ok(uBytesRead == 12, "Read %u bytes, expected %u\n", uBytesRead, 12);
164         ok(memcmp(MemBuf, CmpMem, 12) == 0, "Read returned invalid data!\n");
165 
166         /* check whether the seek pointer has moved correctly */
167         IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
168         ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == uBytesRead, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, uBytesRead);
169     }
170 
171     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
172 
173     hr = IWICStream_Read(pStream, Memory, 10, &uBytesRead);   /* source = dest */
174     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
175     if(SUCCEEDED(hr)) {
176         ok(uBytesRead == 10, "Read %u bytes, expected %u\n", uBytesRead, 10);
177         ok(memcmp(Memory, CmpMem, uBytesRead) == 0, "Read returned invalid data!\n");
178     }
179 
180     IWICStream_Seek(pStream, SeekPos, STREAM_SEEK_SET, NULL);
181 
182     hr = IWICStream_Read(pStream, Memory, 10, &uBytesRead);   /* source and dest overlap */
183     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
184     if(SUCCEEDED(hr)) {
185         ok(uBytesRead == 10, "Read %u bytes, expected %u\n", uBytesRead, 10);
186         ok(memcmp(Memory, CmpMemOverlap, uBytesRead) == 0, "Read returned invalid data!\n");
187     }
188 
189     memcpy(Memory, CmpMem, sizeof(CmpMem));
190 
191     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
192 
193     hr = IWICStream_Read(pStream, Memory, sizeof(Memory) + 10, &uBytesRead);   /* request too many bytes */
194     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
195     if(SUCCEEDED(hr)) {
196         ok(uBytesRead == sizeof(Memory), "Read %u bytes\n", uBytesRead);
197         ok(memcmp(Memory, CmpMem, uBytesRead) == 0, "Read returned invalid data!\n");
198     }
199 
200     hr = IWICStream_Read(pStream, NULL, 1, &uBytesRead);    /* destination buffer = NULL */
201     ok(hr == E_INVALIDARG, "Read returned with %#x, expected %#x\n", hr, E_INVALIDARG);
202 
203     hr = IWICStream_Read(pStream, MemBuf, 0, &uBytesRead);    /* read 0 bytes */
204     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
205 
206     hr = IWICStream_Read(pStream, NULL, 0, &uBytesRead);
207     ok(hr == E_INVALIDARG, "Read returned with %#x, expected %#x\n", hr, E_INVALIDARG);
208 
209     hr = IWICStream_Read(pStream, NULL, 0, NULL);
210     ok(hr == E_INVALIDARG, "Read returned with %#x, expected %#x\n", hr, E_INVALIDARG);
211 
212     hr = IWICStream_Read(pStream, MemBuf, 1, NULL);
213     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
214 
215     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
216     ZeroMemory(MemBuf, sizeof(MemBuf));
217     hr = IWICStream_Read(pStream, MemBuf, sizeof(Memory) + 10, &uBytesRead);
218     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
219     if(SUCCEEDED(hr)) {
220         ok(uBytesRead == sizeof(Memory), "Read %u bytes\n", uBytesRead);
221         ok(memcmp(Memory, CmpMem, 64) == 0, "Read returned invalid data!\n");
222     }
223     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
224 
225 
226     /* Write */
227     MemBuf[0] = CmpMem[0] + 1;
228     MemBuf[1] = CmpMem[1] + 1;
229     MemBuf[2] = CmpMem[2] + 1;
230     hr = IWICStream_Write(pStream, MemBuf, 3, &uBytesWritten);
231     ok(hr == S_OK, "Write returned with %#x, expected %#x\n", hr, S_OK);
232     if(SUCCEEDED(hr)) {
233         ok(uBytesWritten == 3, "Wrote %u bytes, expected %u\n", uBytesWritten, 3);
234         ok(memcmp(MemBuf, Memory, 3) == 0, "Wrote returned invalid data!\n"); /* make sure we're writing directly */
235 
236         /* check whether the seek pointer has moved correctly */
237         IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
238         ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == uBytesWritten, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, uBytesWritten);
239     }
240     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
241 
242     hr = IWICStream_Write(pStream, MemBuf, 0, &uBytesWritten);
243     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
244 
245     /* Restore the original contents of the memory stream. */
246     hr = IWICStream_Write(pStream, CmpMem, sizeof(CmpMem), &uBytesWritten);
247     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
248 
249     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
250 
251     /* Source and destination overlap. */
252     hr = IWICStream_Write(pStream, Memory + 5, 10, &uBytesWritten);
253     ok(hr == S_OK, "Write returned with %#x, expected %#x\n", hr, S_OK);
254     if(SUCCEEDED(hr)) {
255         ok(uBytesWritten == 10, "Wrote %u bytes, expected %u\n", uBytesWritten, 10);
256         ok(memcmp(CmpMemOverlap, Memory, sizeof(CmpMemOverlap)) == 0, "Wrote returned invalid data!\n");
257     }
258 
259     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
260 
261     uBytesWritten = 0xdeadbeef;
262     hr = IWICStream_Write(pStream, NULL, 3, &uBytesWritten);
263     ok(hr == E_INVALIDARG, "Write returned with %#x, expected %#x\n", hr, E_INVALIDARG);
264     ok(uBytesWritten == 0xdeadbeef, "Expected uBytesWritten to be unchanged, got %u\n", uBytesWritten);
265     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
266     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
267 
268     uBytesWritten = 0xdeadbeef;
269     hr = IWICStream_Write(pStream, NULL, 0, &uBytesWritten);
270     ok(hr == E_INVALIDARG, "Write returned with %#x, expected %#x\n", hr, E_INVALIDARG);
271     ok(uBytesWritten == 0xdeadbeef, "Expected uBytesWritten to be unchanged, got %u\n", uBytesWritten);
272     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
273     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
274 
275     uBytesWritten = 0xdeadbeef;
276     hr = IWICStream_Write(pStream, CmpMem, sizeof(Memory) + 10, &uBytesWritten);
277     ok(hr == STG_E_MEDIUMFULL, "Write returned with %#x, expected %#x\n", hr, STG_E_MEDIUMFULL);
278     ok(uBytesWritten == 0xdeadbeef, "Expected uBytesWritten to be unchanged, got %u\n", uBytesWritten);
279     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
280     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
281     IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_SET, NULL);
282 
283 
284     /* SetSize */
285     uNewPos.u.HighPart = 0;
286     uNewPos.u.LowPart = sizeof(Memory) + 10;
287     hr = IWICStream_SetSize(pStream, uNewPos);
288     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
289 
290     uNewPos.u.HighPart = 0;
291     uNewPos.u.LowPart = sizeof(Memory);
292     hr = IWICStream_SetSize(pStream, uNewPos);
293     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
294 
295     uNewPos.u.HighPart = 0;
296     uNewPos.u.LowPart = sizeof(Memory) - 10;
297     hr = IWICStream_SetSize(pStream, uNewPos);
298     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
299 
300     uNewPos.u.HighPart = 0;
301     uNewPos.u.LowPart = 0;
302     hr = IWICStream_SetSize(pStream, uNewPos);
303     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
304 
305     uNewPos.QuadPart = -10;
306     hr = IWICStream_SetSize(pStream, uNewPos);
307     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
308 
309 
310     /* CopyTo */
311     uNewPos.u.HighPart = 0;
312     uNewPos.u.LowPart = 5;
313     hr = IWICStream_CopyTo(pStream, NULL, uNewPos, NULL, NULL);
314     ok(hr == E_NOTIMPL, "CopyTo returned %#x, expected %#x\n", hr, E_NOTIMPL);
315 
316     hr = IWICImagingFactory_CreateStream(pFactory, &pBufStream);
317     ok(hr == S_OK, "CreateStream failed with %#x\n", hr);
318 
319     hr = IWICStream_InitializeFromMemory(pBufStream, Memory, sizeof(Memory));
320     ok(hr == S_OK, "InitializeFromMemory returned with %#x, expected %#x\n", hr, S_OK);
321 
322     hr = IWICStream_CopyTo(pStream, (IStream*)pBufStream, uNewPos, NULL, NULL);
323     ok(hr == E_NOTIMPL, "CopyTo returned %#x, expected %#x\n", hr, E_NOTIMPL);
324     IWICStream_Release(pBufStream);
325 
326 
327     /* Commit */
328     hr = IWICStream_Commit(pStream, STGC_DEFAULT);
329     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
330 
331     hr = IWICStream_Commit(pStream, STGC_OVERWRITE);
332     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
333 
334     hr = IWICStream_Commit(pStream, STGC_ONLYIFCURRENT);
335     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
336 
337     hr = IWICStream_Commit(pStream, STGC_DANGEROUSLYCOMMITMERELYTODISKCACHE);
338     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
339 
340     hr = IWICStream_Commit(pStream, STGC_CONSOLIDATE);
341     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
342 
343 
344     /* Revert */
345     IWICStream_Write(pStream, &MemBuf[5], 6, NULL);
346     hr = IWICStream_Revert(pStream);
347     ok(hr == E_NOTIMPL, "Revert returned %#x, expected %#x\n", hr, E_NOTIMPL);
348     memcpy(Memory, CmpMem, sizeof(Memory));
349 
350 
351     /* LockRegion/UnlockRegion */
352     hr = IWICStream_LockRegion(pStream, uLargeNull, uLargeNull, 0);
353     ok(hr == E_NOTIMPL, "LockRegion returned %#x, expected %#x\n", hr, E_NOTIMPL);
354 
355     hr = IWICStream_UnlockRegion(pStream, uLargeNull, uLargeNull, 0);
356     ok(hr == E_NOTIMPL, "UnlockRegion returned %#x, expected %#x\n", hr, E_NOTIMPL);
357 
358 
359     /* Stat */
360     hr = IWICStream_Stat(pStream, NULL, 0);
361     ok(hr == E_INVALIDARG, "Stat returned %#x, expected %#x\n", hr, E_INVALIDARG);
362 
363     hr = IWICStream_Stat(pStream, &Stats, 0);
364     ok(hr == S_OK, "Stat returned %#x, expected %#x\n", hr, S_OK);
365     ok(Stats.pwcsName == NULL, "Stat returned name %p, expected %p\n", Stats.pwcsName, NULL);
366     ok(Stats.type == STGTY_STREAM, "Stat returned type %d, expected %d\n", Stats.type, STGTY_STREAM);
367     ok(Stats.cbSize.u.HighPart == 0 && Stats.cbSize.u.LowPart == sizeof(Memory), "Stat returned size (%u;%u)\n", Stats.cbSize.u.HighPart, Stats.cbSize.u.LowPart);
368     ok(Stats.mtime.dwHighDateTime == 0 && Stats.mtime.dwLowDateTime == 0, "Stat returned mtime (%u;%u), expected (%u;%u)\n", Stats.mtime.dwHighDateTime, Stats.mtime.dwLowDateTime, 0, 0);
369     ok(Stats.ctime.dwHighDateTime == 0 && Stats.ctime.dwLowDateTime == 0, "Stat returned ctime (%u;%u), expected (%u;%u)\n", Stats.ctime.dwHighDateTime, Stats.ctime.dwLowDateTime, 0, 0);
370     ok(Stats.atime.dwHighDateTime == 0 && Stats.atime.dwLowDateTime == 0, "Stat returned atime (%u;%u), expected (%u;%u)\n", Stats.atime.dwHighDateTime, Stats.atime.dwLowDateTime, 0, 0);
371     ok(Stats.grfMode == 0, "Stat returned access mode %d, expected %d\n", Stats.grfMode, 0);
372     ok(Stats.grfLocksSupported == 0, "Stat returned supported locks %#x, expected %#x\n", Stats.grfLocksSupported, 0);
373     ok(Stats.grfStateBits == 0, "Stat returned state bits %#x, expected %#x\n", Stats.grfStateBits, 0);
374 
375 
376     /* Clone */
377     hr = IWICStream_Clone(pStream, (IStream**)&pBufStream);
378     ok(hr == E_NOTIMPL, "UnlockRegion returned %#x, expected %#x\n", hr, E_NOTIMPL);
379 
380 
381     IWICStream_Release(pStream);
382     IWICImagingFactory_Release(pFactory);
383     CoUninitialize();
384 }
385 
386 static void test_StreamOnStreamRange(void)
387 {
388     IWICImagingFactory *pFactory;
389     IWICStream *pStream, *pSubStream;
390     IStream *CopyStream;
391     const BYTE CmpMem[] = {
392         0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
393         0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
394         0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
395         0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
396     };
397     BYTE Memory[64], MemBuf[64];
398     LARGE_INTEGER LargeNull, LargeInt;
399     ULARGE_INTEGER uLargeNull, uNewPos, uSize;
400     ULONG uBytesRead, uBytesWritten;
401     HRESULT hr;
402     STATSTG Stats;
403 
404     LargeNull.QuadPart = 0;
405     uLargeNull.QuadPart = 0;
406 
407     memcpy(Memory, CmpMem, sizeof(CmpMem));
408 
409     CoInitialize(NULL);
410 
411     hr = CoCreateInstance(&CLSID_WICImagingFactory, NULL, CLSCTX_INPROC_SERVER, &IID_IWICImagingFactory, (void**)&pFactory);
412     if(FAILED(hr)) {
413         skip("CoCreateInstance returned with %#x, expected %#x\n", hr, S_OK);
414         return;
415     }
416 
417     hr = IWICImagingFactory_CreateStream(pFactory, &pStream);
418     ok(hr == S_OK, "CreateStream returned with %#x, expected %#x\n", hr, S_OK);
419     if(FAILED(hr)) {
420         skip("Failed to create stream\n");
421         return;
422     }
423 
424     hr = IWICStream_InitializeFromMemory(pStream, Memory, sizeof(Memory));
425     ok(hr == S_OK, "InitializeFromMemory returned with %#x, expected %#x\n", hr, S_OK);
426 
427     hr = IWICImagingFactory_CreateStream(pFactory, &pSubStream);
428     ok(hr == S_OK, "CreateStream returned with %#x, expected %#x\n", hr, S_OK);
429 
430     uNewPos.QuadPart = 20;
431     uSize.QuadPart = 20;
432     hr = IWICStream_InitializeFromIStreamRegion(pSubStream, (IStream*)pStream, uNewPos, uSize);
433     ok(hr == S_OK, "InitializeFromIStreamRegion returned with %#x, expected %#x\n", hr, S_OK);
434     if(FAILED(hr)) {
435         skip("InitializeFromIStreamRegion unimplemented\n");
436         IWICStream_Release(pSubStream);
437         IWICStream_Release(pStream);
438         IWICImagingFactory_Release(pFactory);
439         CoUninitialize();
440         return;
441     }
442 
443     /* Seek */
444     hr = IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_END, &uNewPos);
445     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 20, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 20);
446     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
447 
448     hr = IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, &uNewPos);
449     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
450     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
451 
452     hr = IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
453     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
454 
455     LargeInt.u.HighPart = 1;
456     LargeInt.u.LowPart = 0;
457     uNewPos.u.HighPart = 0xdeadbeef;
458     uNewPos.u.LowPart = 0xdeadbeef;
459     hr = IWICStream_Seek(pSubStream, LargeInt, STREAM_SEEK_SET, &uNewPos);
460     ok(hr == WINCODEC_ERR_VALUEOUTOFRANGE, "Seek returned with %#x, expected %#x\n", hr, WINCODEC_ERR_VALUEOUTOFRANGE);
461     ok(uNewPos.u.HighPart == 0xdeadbeef && uNewPos.u.LowPart == 0xdeadbeef, "Seek cursor initialized to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0xdeadbeef, 0xdeadbeef);
462     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
463     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
464     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
465 
466     LargeInt.QuadPart = 30;
467     uNewPos.u.HighPart = 0xdeadbeef;
468     uNewPos.u.LowPart = 0xdeadbeef;
469     hr = IWICStream_Seek(pSubStream, LargeInt, STREAM_SEEK_SET, &uNewPos);
470     ok(hr == WINCODEC_ERR_VALUEOUTOFRANGE, "Seek returned with %#x, expected %#x\n", hr, WINCODEC_ERR_VALUEOUTOFRANGE);
471     ok(uNewPos.u.HighPart == 0xdeadbeef && uNewPos.u.LowPart == 0xdeadbeef, "Seek cursor initialized to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0xdeadbeef, 0xdeadbeef);
472     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
473     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
474     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
475 
476     LargeInt.QuadPart = 1;
477     uNewPos.u.HighPart = 0xdeadbeef;
478     uNewPos.u.LowPart = 0xdeadbeef;
479     hr = IWICStream_Seek(pSubStream, LargeInt, STREAM_SEEK_END, &uNewPos);
480     ok(hr == WINCODEC_ERR_VALUEOUTOFRANGE, "Seek returned with %#x, expected %#x\n", hr, WINCODEC_ERR_VALUEOUTOFRANGE);
481     ok(uNewPos.u.HighPart == 0xdeadbeef && uNewPos.u.LowPart == 0xdeadbeef, "Seek cursor initialized to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0xdeadbeef, 0xdeadbeef);
482     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
483     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
484     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
485 
486     LargeInt.QuadPart = -1;
487     hr = IWICStream_Seek(pSubStream, LargeInt, STREAM_SEEK_END, &uNewPos);
488     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
489     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 19, "bSeek cursor moved to position (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart);
490 
491     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, &uNewPos); /* reset seek pointer */
492     LargeInt.QuadPart = -25;
493     uNewPos.u.HighPart = 0xdeadbeef;
494     uNewPos.u.LowPart = 0xdeadbeef;
495     hr = IWICStream_Seek(pSubStream, LargeInt, STREAM_SEEK_END, &uNewPos);
496     ok(hr == WINCODEC_ERR_VALUEOUTOFRANGE,
497        "Seek returned with %#x, expected %#x\n", hr, HRESULT_FROM_WIN32(ERROR_ARITHMETIC_OVERFLOW));
498     ok(uNewPos.u.HighPart == 0xdeadbeef && uNewPos.u.LowPart == 0xdeadbeef, "Seek cursor initialized to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0xdeadbeef, 0xdeadbeef);
499     hr = IWICStream_Seek(pStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
500     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
501     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0); /* remains unchanged */
502     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
503 
504 
505     /* Read */
506     hr = IWICStream_Read(pSubStream, MemBuf, 12, &uBytesRead);
507     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
508     if(SUCCEEDED(hr)) {
509         ok(uBytesRead == 12, "Read %u bytes, expected %u\n", uBytesRead, 12);
510         ok(memcmp(MemBuf, CmpMem+20, 12) == 0, "Read returned invalid data!\n");
511 
512         /* check whether the seek pointer has moved correctly */
513         IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
514         ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == uBytesRead, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, uBytesRead);
515     }
516 
517     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
518 
519     hr = IWICStream_Read(pSubStream, Memory, 10, &uBytesRead);   /* source = dest */
520     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
521     if(SUCCEEDED(hr)) {
522         ok(uBytesRead == 10, "Read %u bytes, expected %u\n", uBytesRead, 10);
523         ok(memcmp(Memory, CmpMem+20, uBytesRead) == 0, "Read returned invalid data!\n");
524     }
525 
526     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
527 
528     hr = IWICStream_Read(pSubStream, Memory, 30, &uBytesRead);   /* request too many bytes */
529     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
530     if(SUCCEEDED(hr)) {
531         ok(uBytesRead == 20, "Read %u bytes\n", uBytesRead);
532         ok(memcmp(Memory, CmpMem+20, uBytesRead) == 0, "Read returned invalid data!\n");
533     }
534 
535     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
536     uBytesRead = 0xdeadbeef;
537     hr = IWICStream_Read(pSubStream, NULL, 1, &uBytesRead);    /* destination buffer = NULL */
538     ok(hr == E_INVALIDARG, "Read returned with %#x, expected %#x\n", hr, E_INVALIDARG);
539     ok(uBytesRead == 0xdeadbeef, "Expected uBytesRead to be unchanged, got %u\n", uBytesRead);
540 
541     hr = IWICStream_Read(pSubStream, MemBuf, 0, &uBytesRead);    /* read 0 bytes */
542     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
543 
544     uBytesRead = 0xdeadbeef;
545     hr = IWICStream_Read(pSubStream, NULL, 0, &uBytesRead);
546     ok(hr == E_INVALIDARG, "Read returned with %#x, expected %#x\n", hr, E_INVALIDARG);
547     ok(uBytesRead == 0xdeadbeef, "Expected uBytesRead to be unchanged, got %u\n", uBytesRead);
548 
549     hr = IWICStream_Read(pSubStream, NULL, 0, NULL);
550     ok(hr == E_INVALIDARG, "Read returned with %#x, expected %#x\n", hr, E_INVALIDARG);
551 
552     hr = IWICStream_Read(pSubStream, MemBuf, 1, NULL);
553     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
554 
555     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
556     ZeroMemory(MemBuf, sizeof(MemBuf));
557     hr = IWICStream_Read(pSubStream, MemBuf, 30, &uBytesRead);
558     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
559     if(SUCCEEDED(hr)) {
560         ok(uBytesRead == 20, "Read %u bytes\n", uBytesRead);
561         ok(memcmp(Memory, CmpMem+20, 20) == 0, "Read returned invalid data!\n");
562     }
563     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
564 
565 
566     /* Write */
567     MemBuf[0] = CmpMem[0] + 1;
568     MemBuf[1] = CmpMem[1] + 1;
569     MemBuf[2] = CmpMem[2] + 1;
570     hr = IWICStream_Write(pSubStream, MemBuf, 3, &uBytesWritten);
571     ok(hr == S_OK, "Write returned with %#x, expected %#x\n", hr, S_OK);
572     if(SUCCEEDED(hr)) {
573         ok(uBytesWritten == 3, "Wrote %u bytes, expected %u\n", uBytesWritten, 3);
574         ok(memcmp(MemBuf, Memory+20, 3) == 0, "Wrote returned invalid data!\n"); /* make sure we're writing directly */
575 
576         /* check whether the seek pointer has moved correctly */
577         IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
578         ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == uBytesWritten, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, uBytesWritten);
579     }
580     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
581 
582     hr = IWICStream_Write(pSubStream, MemBuf, 0, &uBytesWritten);
583     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
584 
585     uBytesWritten = 0xdeadbeef;
586     hr = IWICStream_Write(pSubStream, NULL, 3, &uBytesWritten);
587     ok(hr == E_INVALIDARG, "Write returned with %#x, expected %#x\n", hr, E_INVALIDARG);
588     ok(uBytesWritten == 0xdeadbeef, "Expected uBytesWritten to be unchanged, got %u\n", uBytesWritten);
589     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
590     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
591 
592     uBytesWritten = 0xdeadbeef;
593     hr = IWICStream_Write(pSubStream, NULL, 0, &uBytesWritten);
594     ok(hr == E_INVALIDARG, "Write returned with %#x, expected %#x\n", hr, E_INVALIDARG);
595     ok(uBytesWritten == 0xdeadbeef, "Expected uBytesWritten to be unchanged, got %u\n", uBytesWritten);
596     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
597     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
598 
599     hr = IWICStream_Write(pSubStream, CmpMem, 30, &uBytesWritten);
600     ok(hr == S_OK, "Write returned with %#x, expected %#x\n", hr, STG_E_MEDIUMFULL);
601     ok(uBytesWritten == 20, "Wrote %u bytes, expected %u\n", uBytesWritten, 0);
602     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
603     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == uBytesWritten, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, uBytesWritten);
604     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
605 
606 
607     /* SetSize */
608     uNewPos.u.HighPart = 0;
609     uNewPos.u.LowPart = sizeof(Memory) + 10;
610     hr = IWICStream_SetSize(pSubStream, uNewPos);
611     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
612 
613     uNewPos.u.HighPart = 0;
614     uNewPos.u.LowPart = sizeof(Memory);
615     hr = IWICStream_SetSize(pSubStream, uNewPos);
616     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
617 
618     uNewPos.u.HighPart = 0;
619     uNewPos.u.LowPart = sizeof(Memory) - 10;
620     hr = IWICStream_SetSize(pSubStream, uNewPos);
621     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
622 
623     uNewPos.u.HighPart = 0;
624     uNewPos.u.LowPart = 0;
625     hr = IWICStream_SetSize(pSubStream, uNewPos);
626     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
627 
628     uNewPos.QuadPart = -10;
629     hr = IWICStream_SetSize(pSubStream, uNewPos);
630     ok(hr == E_NOTIMPL, "SetSize returned %#x, expected %#x\n", hr, E_NOTIMPL);
631 
632 
633     /* CopyTo */
634     uNewPos.u.HighPart = 0;
635     uNewPos.u.LowPart = 30;
636     hr = IWICStream_CopyTo(pSubStream, NULL, uNewPos, NULL, NULL);
637     ok(hr == E_NOTIMPL, "CopyTo returned %#x, expected %#x\n", hr, E_NOTIMPL);
638 
639     hr = CreateStreamOnHGlobal(NULL, TRUE, &CopyStream);
640     ok(hr == S_OK, "CreateStream failed with %#x\n", hr);
641 
642     hr = IWICStream_CopyTo(pSubStream, CopyStream, uNewPos, NULL, NULL);
643     ok(hr == E_NOTIMPL, "CopyTo returned %#x, expected %#x\n", hr, E_NOTIMPL);
644     IStream_Release(CopyStream);
645 
646 
647     /* Commit */
648     hr = IWICStream_Commit(pSubStream, STGC_DEFAULT);
649     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
650 
651     hr = IWICStream_Commit(pSubStream, STGC_OVERWRITE);
652     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
653 
654     hr = IWICStream_Commit(pSubStream, STGC_ONLYIFCURRENT);
655     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
656 
657     hr = IWICStream_Commit(pSubStream, STGC_DANGEROUSLYCOMMITMERELYTODISKCACHE);
658     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
659 
660     hr = IWICStream_Commit(pSubStream, STGC_CONSOLIDATE);
661     ok(hr == E_NOTIMPL, "Commit returned %#x, expected %#x\n", hr, E_NOTIMPL);
662 
663 
664     /* Revert */
665     IWICStream_Write(pSubStream, &MemBuf[5], 6, NULL);
666     hr = IWICStream_Revert(pSubStream);
667     ok(hr == E_NOTIMPL, "Revert returned %#x, expected %#x\n", hr, E_NOTIMPL);
668     memcpy(Memory, CmpMem, sizeof(Memory));
669 
670 
671     /* LockRegion/UnlockRegion */
672     hr = IWICStream_LockRegion(pSubStream, uLargeNull, uLargeNull, 0);
673     ok(hr == E_NOTIMPL, "LockRegion returned %#x, expected %#x\n", hr, E_NOTIMPL);
674 
675     hr = IWICStream_UnlockRegion(pSubStream, uLargeNull, uLargeNull, 0);
676     ok(hr == E_NOTIMPL, "UnlockRegion returned %#x, expected %#x\n", hr, E_NOTIMPL);
677 
678 
679     /* Stat */
680     hr = IWICStream_Stat(pSubStream, NULL, 0);
681     ok(hr == E_INVALIDARG, "Stat returned %#x, expected %#x\n", hr, E_INVALIDARG);
682 
683     hr = IWICStream_Stat(pSubStream, &Stats, 0);
684     ok(hr == S_OK, "Stat returned %#x, expected %#x\n", hr, S_OK);
685     ok(Stats.pwcsName == NULL, "Stat returned name %p, expected %p\n", Stats.pwcsName, NULL);
686     ok(Stats.type == STGTY_STREAM, "Stat returned type %d, expected %d\n", Stats.type, STGTY_STREAM);
687     ok(Stats.cbSize.u.HighPart == 0 && Stats.cbSize.u.LowPart == 20, "Stat returned size (%u;%u)\n", Stats.cbSize.u.HighPart, Stats.cbSize.u.LowPart);
688     ok(Stats.mtime.dwHighDateTime == 0 && Stats.mtime.dwLowDateTime == 0, "Stat returned mtime (%u;%u), expected (%u;%u)\n", Stats.mtime.dwHighDateTime, Stats.mtime.dwLowDateTime, 0, 0);
689     ok(Stats.ctime.dwHighDateTime == 0 && Stats.ctime.dwLowDateTime == 0, "Stat returned ctime (%u;%u), expected (%u;%u)\n", Stats.ctime.dwHighDateTime, Stats.ctime.dwLowDateTime, 0, 0);
690     ok(Stats.atime.dwHighDateTime == 0 && Stats.atime.dwLowDateTime == 0, "Stat returned atime (%u;%u), expected (%u;%u)\n", Stats.atime.dwHighDateTime, Stats.atime.dwLowDateTime, 0, 0);
691     ok(Stats.grfMode == 0, "Stat returned access mode %d, expected %d\n", Stats.grfMode, 0);
692     ok(Stats.grfLocksSupported == 0, "Stat returned supported locks %#x, expected %#x\n", Stats.grfLocksSupported, 0);
693     ok(Stats.grfStateBits == 0, "Stat returned state bits %#x, expected %#x\n", Stats.grfStateBits, 0);
694 
695 
696     /* Clone */
697     hr = IWICStream_Clone(pSubStream, &CopyStream);
698     ok(hr == E_NOTIMPL, "Clone returned %#x, expected %#x\n", hr, E_NOTIMPL);
699 
700 
701     IWICStream_Release(pSubStream);
702 
703 
704     /* Recreate, this time larger than the original. */
705     hr = IWICImagingFactory_CreateStream(pFactory, &pSubStream);
706     ok(hr == S_OK, "CreateStream returned with %#x, expected %#x\n", hr, S_OK);
707 
708     uNewPos.QuadPart = 48;
709     uSize.QuadPart = 32;
710     hr = IWICStream_InitializeFromIStreamRegion(pSubStream, (IStream*)pStream, uNewPos, uSize);
711     ok(hr == S_OK, "InitializeFromMemory returned with %#x, expected %#x\n", hr, S_OK);
712 
713     hr = IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_END, &uNewPos);
714     ok(hr == S_OK, "Seek returned with %#x, expected %#x\n", hr, S_OK);
715     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 16, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 16);
716 
717     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
718     hr = IWICStream_Read(pSubStream, Memory, 48, &uBytesRead);
719     ok(hr == S_OK, "Read returned with %#x, expected %#x\n", hr, S_OK);
720     if(SUCCEEDED(hr)) {
721         ok(uBytesRead == 16, "Read %u bytes\n", uBytesRead);
722         ok(memcmp(Memory, CmpMem+48, uBytesRead) == 0, "Read returned invalid data!\n");
723     }
724 
725     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_SET, NULL);
726     uBytesWritten = 0xdeadbeef;
727     hr = IWICStream_Write(pSubStream, CmpMem, 32, &uBytesWritten);
728     ok(hr == STG_E_MEDIUMFULL, "Write returned with %#x, expected %#x\n", hr, STG_E_MEDIUMFULL);
729     ok(uBytesWritten == 0xdeadbeef, "Expected uBytesWritten to be unchanged, got %u\n", uBytesWritten);
730     IWICStream_Seek(pSubStream, LargeNull, STREAM_SEEK_CUR, &uNewPos);
731     ok(uNewPos.u.HighPart == 0 && uNewPos.u.LowPart == 0, "Seek cursor moved to position (%u;%u), expected (%u;%u)\n", uNewPos.u.HighPart, uNewPos.u.LowPart, 0, 0);
732 
733 
734     IWICStream_Release(pSubStream);
735     IWICStream_Release(pStream);
736     IWICImagingFactory_Release(pFactory);
737     CoUninitialize();
738 }
739 
740 START_TEST(stream)
741 {
742     test_StreamOnMemory();
743     test_StreamOnStreamRange();
744 }
745