user32: Reimplement 16-bit clipboard functions on top of the 32-bit ones.
[wine] / dlls / ole32 / tests / hglobalstream.c
1 /*
2  * Stream on HGLOBAL Tests
3  *
4  * Copyright 2006 Robert Shearman (for CodeWeavers)
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  */
20
21 #define COBJMACROS
22
23 #include <stdarg.h>
24
25 #include "windef.h"
26 #include "winbase.h"
27 #include "objbase.h"
28
29 #include "wine/test.h"
30
31 #define ok_ole_success(hr, func) ok(hr == S_OK, func " failed with error 0x%08x\n", hr)
32
33 static char const * const *expected_method_list;
34
35 #define CHECK_EXPECTED_METHOD(method_name) \
36 do { \
37     ok(*expected_method_list != NULL, "Extra method %s called\n", method_name); \
38         if (*expected_method_list) \
39         { \
40             ok(!strcmp(*expected_method_list, method_name), "Expected %s to be called instead of %s\n", \
41                *expected_method_list, method_name); \
42                    expected_method_list++; \
43         } \
44 } while(0)
45
46 static void test_streamonhglobal(IStream *pStream)
47 {
48     const char data[] = "Test String";
49     ULARGE_INTEGER ull;
50     LARGE_INTEGER ll;
51     char buffer[128];
52     ULONG read;
53     STATSTG statstg;
54     HRESULT hr;
55
56     ull.QuadPart = sizeof(data);
57     hr = IStream_SetSize(pStream, ull);
58     ok_ole_success(hr, "IStream_SetSize");
59
60     hr = IStream_Write(pStream, data, sizeof(data), NULL);
61     ok_ole_success(hr, "IStream_Write");
62
63     ll.QuadPart = 0;
64     hr = IStream_Seek(pStream, ll, STREAM_SEEK_SET, NULL);
65     ok_ole_success(hr, "IStream_Seek");
66
67     /* should return S_OK, not S_FALSE */
68     hr = IStream_Read(pStream, buffer, sizeof(buffer), &read);
69     ok_ole_success(hr, "IStream_Read");
70     ok(read == sizeof(data), "IStream_Read returned read %d\n", read);
71
72     /* ignores HighPart */
73     ull.u.HighPart = -1;
74     ull.u.LowPart = 0;
75     hr = IStream_SetSize(pStream, ull);
76     ok_ole_success(hr, "IStream_SetSize");
77
78     /* ignores HighPart */
79     ll.u.HighPart = -1;
80     ll.u.LowPart = 0;
81     hr = IStream_Seek(pStream, ll, STREAM_SEEK_SET, &ull);
82     todo_wine
83     ok_ole_success(hr, "IStream_Seek");
84     ok(ull.u.LowPart == 0, "should have set LowPart to 0 instead of %d\n", ull.u.LowPart);
85     todo_wine
86     ok(ull.u.HighPart == 0, "should have set HighPart to 0 instead of %d\n", ull.u.HighPart);
87
88     /* ignores HighPart */
89     ll.u.HighPart = -1;
90     ll.u.LowPart = 0;
91     hr = IStream_Seek(pStream, ll, STREAM_SEEK_CUR, NULL);
92     todo_wine
93     ok_ole_success(hr, "IStream_Seek");
94
95     hr = IStream_Commit(pStream, STGC_DEFAULT);
96     ok_ole_success(hr, "IStream_Commit");
97
98     hr = IStream_Revert(pStream);
99     ok_ole_success(hr, "IStream_Revert");
100
101     hr = IStream_LockRegion(pStream, ull, ull, LOCK_WRITE);
102     ok(hr == STG_E_INVALIDFUNCTION, "IStream_LockRegion should have returned STG_E_INVALIDFUNCTION instead of 0x%08x\n", hr);
103
104     hr = IStream_Stat(pStream, &statstg, STATFLAG_DEFAULT);
105     ok_ole_success(hr, "IStream_Stat");
106     ok(statstg.type == STGTY_STREAM, "statstg.type should have been STGTY_STREAM instead of %d\n", statstg.type);
107
108     /* test OOM condition */
109     ull.u.HighPart = -1;
110     ull.u.LowPart = -1;
111     hr = IStream_SetSize(pStream, ull);
112     ok(hr == E_OUTOFMEMORY || broken(hr == S_OK), /* win9x */
113        "IStream_SetSize with large size should have returned E_OUTOFMEMORY instead of 0x%08x\n", hr);
114 }
115
116 static HRESULT WINAPI TestStream_QueryInterface(IStream *iface, REFIID riid, void **ppv)
117 {
118     if (IsEqualIID(riid, &IID_IUnknown) ||
119         IsEqualIID(riid, &IID_ISequentialStream) ||
120         IsEqualIID(riid, &IID_IStream))
121     {
122         *ppv = iface;
123         IUnknown_AddRef(iface);
124         return S_OK;
125     }
126     *ppv = NULL;
127     return E_NOINTERFACE;
128 }
129
130 static ULONG WINAPI TestStream_AddRef(IStream *iface)
131 {
132     return 2;
133 }
134
135 static ULONG WINAPI TestStream_Release(IStream *iface)
136 {
137     return 1;
138 }
139
140 static HRESULT WINAPI TestStream_Read(IStream *iface, void *pv, ULONG cb, ULONG *pcbRead)
141 {
142     CHECK_EXPECTED_METHOD("TestStream_Read");
143     return E_NOTIMPL;
144 }
145
146 static HRESULT WINAPI TestStream_Write(IStream *iface, const void *pv, ULONG cb, ULONG *pcbWritten)
147 {
148     CHECK_EXPECTED_METHOD("TestStream_Write");
149     *pcbWritten = 5;
150     return S_OK;
151 }
152
153 static HRESULT WINAPI TestStream_Seek(IStream *iface, LARGE_INTEGER dlibMove, DWORD dwOrigin, ULARGE_INTEGER *plibNewPosition)
154 {
155     CHECK_EXPECTED_METHOD("TestStream_Seek");
156     return E_NOTIMPL;
157 }
158
159 static HRESULT WINAPI TestStream_SetSize(IStream *iface, ULARGE_INTEGER libNewSize)
160 {
161     CHECK_EXPECTED_METHOD("TestStream_SetSize");
162     return E_NOTIMPL;
163 }
164
165 static HRESULT WINAPI TestStream_CopyTo(IStream *iface, IStream *pStream, ULARGE_INTEGER cb, ULARGE_INTEGER *pcbRead, ULARGE_INTEGER *pcbWritten)
166 {
167     CHECK_EXPECTED_METHOD("TestStream_CopyTo");
168     return E_NOTIMPL;
169 }
170
171 static HRESULT WINAPI TestStream_Commit(IStream *iface, DWORD grfCommitFlags)
172 {
173     CHECK_EXPECTED_METHOD("TestStream_Commit");
174     return E_NOTIMPL;
175 }
176
177 static HRESULT WINAPI TestStream_Revert(IStream *iface)
178 {
179     CHECK_EXPECTED_METHOD("TestStream_Revert");
180     return E_NOTIMPL;
181 }
182
183 static HRESULT WINAPI TestStream_LockRegion(IStream *iface, ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
184 {
185     CHECK_EXPECTED_METHOD("TestStream_LockRegion");
186     return E_NOTIMPL;
187 }
188
189 static HRESULT WINAPI TestStream_UnlockRegion(IStream *iface, ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
190 {
191     CHECK_EXPECTED_METHOD("TestStream_UnlockRegion");
192     return E_NOTIMPL;
193 }
194
195 static HRESULT WINAPI TestStream_Stat(IStream *iface, STATSTG *pstatstg, DWORD grfStatFlag)
196 {
197     CHECK_EXPECTED_METHOD("TestStream_Stat");
198     return E_NOTIMPL;
199 }
200
201 static HRESULT WINAPI TestStream_Clone(IStream *iface, IStream **pStream)
202 {
203     CHECK_EXPECTED_METHOD("TestStream_Clone");
204     return E_NOTIMPL;
205 }
206
207 static /*const*/ IStreamVtbl StreamVtbl =
208 {
209     TestStream_QueryInterface,
210     TestStream_AddRef,
211     TestStream_Release,
212     TestStream_Read,
213     TestStream_Write,
214     TestStream_Seek,
215     TestStream_SetSize,
216     TestStream_CopyTo,
217     TestStream_Commit,
218     TestStream_Revert,
219     TestStream_LockRegion,
220     TestStream_UnlockRegion,
221     TestStream_Stat,
222     TestStream_Clone
223 };
224
225 static IStream Test_Stream = { &StreamVtbl };
226
227 static void test_copyto(void)
228 {
229     IStream *pStream, *pStream2;
230     HRESULT hr = CreateStreamOnHGlobal(NULL, TRUE, &pStream);
231     static const char szHello[] = "Hello";
232     ULARGE_INTEGER cb;
233     static const char *methods_copyto[] =
234     {
235         "TestStream_Write",
236         NULL
237     };
238     ULONG written;
239     ULARGE_INTEGER ullRead;
240     ULARGE_INTEGER ullWritten;
241     ULARGE_INTEGER libNewPosition;
242     static const LARGE_INTEGER llZero;
243     char buffer[15];
244
245     expected_method_list = methods_copyto;
246
247     hr = IStream_Write(pStream, szHello, sizeof(szHello), &written);
248     ok_ole_success(hr, "IStream_Write");
249     ok(written == sizeof(szHello), "only %d bytes written\n", written);
250
251     hr = IStream_Seek(pStream, llZero, STREAM_SEEK_SET, NULL);
252     ok_ole_success(hr, "IStream_Seek");
253
254     cb.QuadPart = sizeof(szHello);
255     hr = IStream_CopyTo(pStream, &Test_Stream, cb, &ullRead, &ullWritten);
256     ok(ullWritten.QuadPart == 5, "ullWritten was %d instead\n", (ULONG)ullWritten.QuadPart);
257     ok(ullRead.QuadPart == sizeof(szHello), "only %d bytes read\n", (ULONG)ullRead.QuadPart);
258     ok_ole_success(hr, "IStream_CopyTo");
259
260     ok(!*expected_method_list, "Method sequence starting from %s not called\n", *expected_method_list);
261
262     hr = IStream_Clone(pStream, &pStream2);
263     ok_ole_success(hr, "IStream_Clone");
264
265     hr = IStream_Seek(pStream2, llZero, STREAM_SEEK_CUR, &libNewPosition);
266     ok_ole_success(hr, "IStream_Seek");
267     ok(libNewPosition.QuadPart == sizeof(szHello), "libNewPosition wasn't set correctly for the cloned stream\n");
268
269     hr = IStream_Seek(pStream2, llZero, STREAM_SEEK_SET, NULL);
270     ok_ole_success(hr, "IStream_Seek");
271
272     hr = IStream_Read(pStream2, buffer, sizeof(buffer), NULL);
273     ok_ole_success(hr, "IStream_Read");
274     ok(!strcmp(buffer, szHello), "read data \"%s\" didn't match originally written data\n", buffer);
275
276     IStream_Release(pStream2);
277     IStream_Release(pStream);
278 }
279
280 static void test_freed_hglobal(void)
281 {
282     static const char teststring[] = "this is a test string";
283     HRESULT hr;
284     IStream *pStream;
285     HGLOBAL hglobal;
286     char *p;
287     char buffer[sizeof(teststring) + 8];
288     ULARGE_INTEGER ull;
289     ULONG read, written;
290
291     hglobal = GlobalAlloc(GMEM_DDESHARE|GMEM_NODISCARD|GMEM_MOVEABLE, strlen(teststring) + 1);
292     ok(hglobal != NULL, "GlobalAlloc failed with error %d\n", GetLastError());
293     p = GlobalLock(hglobal);
294     strcpy(p, teststring);
295     GlobalUnlock(hglobal);
296
297     hr = CreateStreamOnHGlobal(hglobal, FALSE, &pStream);
298     ok_ole_success(hr, "CreateStreamOnHGlobal");
299
300     hr = IStream_Read(pStream, buffer, sizeof(buffer), &read);
301     ok_ole_success(hr, "IStream_Read");
302     ok(!strcmp(buffer, teststring), "buffer data %s differs\n", buffer);
303     ok(read == sizeof(teststring) ||
304        broken(read == ((sizeof(teststring) + 3) & ~3)), /* win9x rounds the size */
305        "read should be sizeof(teststring) instead of %d\n", read);
306
307     GlobalFree(hglobal);
308
309     memset(buffer, 0, sizeof(buffer));
310     read = -1;
311     hr = IStream_Read(pStream, buffer, sizeof(buffer), &read);
312     ok_ole_success(hr, "IStream_Read");
313     ok(buffer[0] == 0, "buffer data should be untouched\n");
314     ok(read == 0, "read should be 0 instead of %d\n", read);
315
316     ull.QuadPart = sizeof(buffer);
317     hr = IStream_SetSize(pStream, ull);
318     ok(hr == E_OUTOFMEMORY, "IStream_SetSize with invalid HGLOBAL should return E_OUTOFMEMORY instead of 0x%08x\n", hr);
319
320     hr = IStream_Write(pStream, buffer, sizeof(buffer), &written);
321     ok(hr == E_OUTOFMEMORY, "IStream_Write with invalid HGLOBAL should return E_OUTOFMEMORY instead of 0x%08x\n", hr);
322     ok(written == 0, "written should be 0 instead of %d\n", written);
323
324     IStream_Release(pStream);
325 }
326
327 START_TEST(hglobalstream)
328 {
329     HRESULT hr;
330     IStream *pStream;
331
332     hr = CreateStreamOnHGlobal(NULL, TRUE, &pStream);
333     ok_ole_success(hr, "CreateStreamOnHGlobal");
334
335     test_streamonhglobal(pStream);
336     IStream_Release(pStream);
337     test_copyto();
338     test_freed_hglobal();
339 }