/* Unit test suite for SHLWAPI ShCreateStreamOnFile functions. * * Copyright 2008 Reece H. Dunn * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #define COBJMACROS #include #include #include "wine/test.h" #include "windef.h" #include "winbase.h" #include "objbase.h" #include "shlwapi.h" static void test_IStream_invalid_operations(IStream * stream, DWORD mode) { HRESULT ret; IStream * clone; ULONG refcount; ULARGE_INTEGER uzero; ULARGE_INTEGER uret; LARGE_INTEGER zero; ULONG count; char data[256]; U(uzero).HighPart = 0; U(uzero).LowPart = 0; U(uret).HighPart = 0; U(uret).LowPart = 0; U(zero).HighPart = 0; U(zero).LowPart = 0; /* IStream::Read */ /* IStream_Read from the COBJMACROS is undefined by shlwapi.h, replaced by the IStream_Read helper function. */ ret = stream->lpVtbl->Read(stream, NULL, 0, &count); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); ret = stream->lpVtbl->Read(stream, data, 5, NULL); ok(ret == S_FALSE || ret == S_OK, "expected S_FALSE or S_OK, got 0x%08x\n", ret); ret = stream->lpVtbl->Read(stream, data, 0, NULL); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); ret = stream->lpVtbl->Read(stream, data, 3, &count); ok(ret == S_FALSE || ret == S_OK, "expected S_FALSE or S_OK, got 0x%08x\n", ret); /* IStream::Write */ /* IStream_Write from the COBJMACROS is undefined by shlwapi.h, replaced by the IStream_Write helper function. */ ret = stream->lpVtbl->Write(stream, NULL, 0, &count); if (mode == STGM_READ) { ok(ret == STG_E_ACCESSDENIED /* XP */ || broken(ret == S_OK) /* Win2000 + IE5 */, "expected STG_E_ACCESSDENIED, got 0x%08x\n", ret); } else ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); strcpy(data, "Hello"); ret = stream->lpVtbl->Write(stream, data, 5, NULL); if (mode == STGM_READ) ok(ret == STG_E_ACCESSDENIED, "expected STG_E_ACCESSDENIED, got 0x%08x\n", ret); else ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); strcpy(data, "Hello"); ret = stream->lpVtbl->Write(stream, data, 0, NULL); if (mode == STGM_READ) ok(ret == STG_E_ACCESSDENIED, "expected STG_E_ACCESSDENIED, got 0x%08x\n", ret); else ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); strcpy(data, "Hello"); ret = stream->lpVtbl->Write(stream, data, 0, &count); if (mode == STGM_READ) ok(ret == STG_E_ACCESSDENIED, "expected STG_E_ACCESSDENIED, got 0x%08x\n", ret); else ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); strcpy(data, "Hello"); ret = stream->lpVtbl->Write(stream, data, 3, &count); if (mode == STGM_READ) ok(ret == STG_E_ACCESSDENIED, "expected STG_E_ACCESSDENIED, got 0x%08x\n", ret); else ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); /* IStream::Seek */ ret = IStream_Seek(stream, zero, STREAM_SEEK_SET, NULL); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); ret = IStream_Seek(stream, zero, 20, NULL); ok(ret == E_INVALIDARG, "expected E_INVALIDARG, got 0x%08x\n", ret); /* IStream::CopyTo */ ret = IStream_CopyTo(stream, NULL, uzero, &uret, &uret); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); clone = NULL; ret = IStream_CopyTo(stream, clone, uzero, &uret, &uret); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); ret = IStream_CopyTo(stream, stream, uzero, &uret, &uret); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); ret = IStream_CopyTo(stream, stream, uzero, &uret, NULL); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); ret = IStream_CopyTo(stream, stream, uzero, NULL, &uret); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); /* IStream::Commit */ ret = IStream_Commit(stream, STGC_DEFAULT); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); /* IStream::Revert */ ret = IStream_Revert(stream); ok(ret == E_NOTIMPL, "expected E_NOTIMPL, got 0x%08x\n", ret); /* IStream::LockRegion */ ret = IStream_LockRegion(stream, uzero, uzero, 0); ok(ret == E_NOTIMPL /* XP */ || ret == S_OK /* Vista */, "expected E_NOTIMPL or S_OK, got 0x%08x\n", ret); /* IStream::UnlockRegion */ if (ret == E_NOTIMPL) /* XP */ { ret = IStream_UnlockRegion(stream, uzero, uzero, 0); ok(ret == E_NOTIMPL, "expected E_NOTIMPL, got 0x%08x\n", ret); } else /* Vista */ { ret = IStream_UnlockRegion(stream, uzero, uzero, 0); ok(ret == S_OK, "expected S_OK, got 0x%08x\n", ret); ret = IStream_UnlockRegion(stream, uzero, uzero, 0); ok(ret == STG_E_LOCKVIOLATION, "expected STG_E_LOCKVIOLATION, got 0x%08x\n", ret); } /* IStream::Stat */ ret = IStream_Stat(stream, NULL, 0); ok(ret == STG_E_INVALIDPOINTER, "expected STG_E_INVALIDPOINTER or E_NOTIMPL, got 0x%08x\n", ret); /* IStream::Clone */ /* Passing a NULL pointer for the second IStream::Clone param crashes on Win7 */ clone = NULL; ret = IStream_Clone(stream, &clone); ok(ret == E_NOTIMPL, "expected E_NOTIMPL, got 0x%08x\n", ret); ok(clone == NULL, "expected a NULL IStream object, got %p\n", stream); if (clone) { refcount = IStream_Release(clone); ok(refcount == 0, "expected 0, got %d\n", refcount); } } static void test_stream_read_write(IStream *stream, DWORD mode) { static const LARGE_INTEGER start; HRESULT ret; unsigned char buf[16]; DWORD written, count; STATSTG statstg; /* IStream_Read/Write from the COBJMACROS is undefined by shlwapi.h */ written = 0xdeadbeaf; ret = stream->lpVtbl->Write(stream, "\x5e\xa7", 2, &written); if (mode == STGM_WRITE || mode == STGM_READWRITE) { ok(ret == S_OK, "IStream_Write error %#x (access %#x)\n", ret, mode); ok(written == 2, "expected 2, got %u\n", written); } else { ok(ret == STG_E_ACCESSDENIED || broken(ret == S_OK) /* win2000 */, "expected STG_E_ACCESSDENIED, got %#x (access %#x)\n", ret, mode); ok(written == 0xdeadbeaf || broken(written == 2) /* win2000 */, "expected 0xdeadbeaf, got %#x\n", written); written = 0; if (ret == S_OK) return; /* no point in further testing */ } ret = stream->lpVtbl->Seek(stream, start, STREAM_SEEK_SET, NULL); ok(ret == S_OK, "Seek error %#x\n", ret); count = 0xdeadbeaf; ret = stream->lpVtbl->Read(stream, buf, 2, &count); if (written != 0) { ok(ret == S_OK || broken(ret == S_FALSE) /* win2000 */, "IStream_Read error %#x (access %#x, written %u)\n", ret, mode, written); if (ret == S_OK && (mode == STGM_WRITE || mode == STGM_READWRITE)) { ok(count == 2, "expected 2, got %u\n", count); ok(buf[0] == 0x5e && buf[1] == 0xa7, "expected 5ea7, got %02x%02x\n", buf[0], buf[1]); } else ok(count == 0, "expected 0, got %u\n", count); } else { ok(ret == S_FALSE, "expected S_FALSE, got %#x (access %#x, written %u)\n", ret, mode, written); ok(count == 0, "expected 0, got %u\n", count); } ret = stream->lpVtbl->Seek(stream, start, STREAM_SEEK_SET, NULL); ok(ret == S_OK, "Seek error %#x\n", ret); count = 0xdeadbeaf; ret = stream->lpVtbl->Read(stream, buf, 0, &count); ok(ret == S_OK, "IStream_Read error %#x (access %#x, written %u)\n", ret, mode, written); ok(count == 0, "expected 0, got %u\n", count); count = 0xdeadbeaf; ret = stream->lpVtbl->Read(stream, buf, sizeof(buf), &count); ok(ret == S_FALSE, "expected S_FALSE, got %#x (access %#x, written %u)\n", ret, mode, written); ok(count == written, "expected %u, got %u\n", written, count); if (count) ok(buf[0] == 0x5e && buf[1] == 0xa7, "expected 5ea7, got %02x%02x\n", buf[0], buf[1]); memset(&statstg, 0xff, sizeof(statstg)); ret = IStream_Stat(stream, &statstg, 0); ok(ret == S_OK, "Stat failed, hr %#x.\n", ret); ok(statstg.pwcsName != NULL, "Unexpected name %s.\n", wine_dbgstr_w(statstg.pwcsName)); CoTaskMemFree(statstg.pwcsName); memset(&statstg, 0xff, sizeof(statstg)); ret = IStream_Stat(stream, &statstg, STATFLAG_NONAME); ok(ret == S_OK, "Stat failed, hr %#x.\n", ret); ok(statstg.pwcsName == NULL, "Unexpected name %s.\n", wine_dbgstr_w(statstg.pwcsName)); } static void test_stream_qi(IStream *stream) { IUnknown *unk; HRESULT hr; hr = IStream_QueryInterface(stream, &IID_IStream, (void **)&unk); ok(SUCCEEDED(hr), "Failed to get IStream interface, hr %#x.\n", hr); IUnknown_Release(unk); unk = NULL; hr = IStream_QueryInterface(stream, &IID_ISequentialStream, (void **)&unk); ok(SUCCEEDED(hr) || broken(hr == E_NOINTERFACE) /* XP */, "Failed to get ISequentialStream interface, hr %#x.\n", hr); if (unk) IUnknown_Release(unk); hr = IStream_QueryInterface(stream, &IID_IUnknown, (void **)&unk); ok(SUCCEEDED(hr), "Failed to get IUnknown interface, hr %#x.\n", hr); IUnknown_Release(unk); } static void test_SHCreateStreamOnFileA(DWORD mode, DWORD stgm) { IStream * stream; HRESULT ret; ULONG refcount; char test_file[MAX_PATH]; static const CHAR testA_txt[] = "\\testA.txt"; trace("SHCreateStreamOnFileA: testing mode %d, STGM flags %08x\n", mode, stgm); /* Don't used a fixed path for the testA.txt file */ GetTempPathA(MAX_PATH, test_file); lstrcatA(test_file, testA_txt); /* invalid arguments */ stream = NULL; ret = SHCreateStreamOnFileA(NULL, mode | stgm, &stream); if (ret == E_INVALIDARG) /* Win98 SE */ { win_skip("Not supported\n"); return; } ok(ret == HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND) /* NT */ || ret == HRESULT_FROM_WIN32(ERROR_BAD_PATHNAME) /* 9x */, "SHCreateStreamOnFileA: expected HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND) " "or HRESULT_FROM_WIN32(ERROR_BAD_PATHNAME), got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileA: expected a NULL IStream object, got %p\n", stream); if (0) /* This test crashes on WinXP SP2 */ { ret = SHCreateStreamOnFileA(test_file, mode | stgm, NULL); ok(ret == E_INVALIDARG, "SHCreateStreamOnFileA: expected E_INVALIDARG, got 0x%08x\n", ret); } stream = NULL; ret = SHCreateStreamOnFileA(test_file, mode | STGM_CONVERT | stgm, &stream); ok(ret == E_INVALIDARG, "SHCreateStreamOnFileA: expected E_INVALIDARG, got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileA: expected a NULL IStream object, got %p\n", stream); stream = NULL; ret = SHCreateStreamOnFileA(test_file, mode | STGM_DELETEONRELEASE | stgm, &stream); ok(ret == E_INVALIDARG, "SHCreateStreamOnFileA: expected E_INVALIDARG, got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileA: expected a NULL IStream object, got %p\n", stream); stream = NULL; ret = SHCreateStreamOnFileA(test_file, mode | STGM_TRANSACTED | stgm, &stream); ok(ret == E_INVALIDARG, "SHCreateStreamOnFileA: expected E_INVALIDARG, got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileA: expected a NULL IStream object, got %p\n", stream); /* file does not exist */ stream = NULL; ret = SHCreateStreamOnFileA(test_file, mode | STGM_FAILIFTHERE | stgm, &stream); ok(ret == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND), "SHCreateStreamOnFileA: expected HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND), got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileA: expected a NULL IStream object, got %p\n", stream); stream = NULL; ret = SHCreateStreamOnFileA(test_file, mode | STGM_CREATE | stgm, &stream); ok(ret == S_OK, "SHCreateStreamOnFileA: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileA: expected a valid IStream object, got NULL\n"); if (stream) { test_stream_qi(stream); test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileA: expected 0, got %d\n", refcount); } /* NOTE: don't delete the file, as it will be used for the file exists tests. */ /* file exists */ stream = NULL; ret = SHCreateStreamOnFileA(test_file, mode | STGM_FAILIFTHERE | stgm, &stream); ok(ret == S_OK, "SHCreateStreamOnFileA: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileA: expected a valid IStream object, got NULL\n"); if (stream) { test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileA: expected 0, got %d\n", refcount); } stream = NULL; ret = SHCreateStreamOnFileA(test_file, mode | STGM_CREATE | stgm, &stream); ok(ret == S_OK, "SHCreateStreamOnFileA: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileA: expected a valid IStream object, got NULL\n"); if (stream) { BOOL delret; test_stream_read_write(stream, mode); test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileA: expected 0, got %d\n", refcount); delret = DeleteFileA(test_file); ok(delret, "SHCreateStreamOnFileA: could not delete file '%s', got error %d\n", test_file, GetLastError()); } } static void test_SHCreateStreamOnFileW(DWORD mode, DWORD stgm) { IStream * stream; HRESULT ret; ULONG refcount; WCHAR test_file[MAX_PATH]; CHAR test_fileA[MAX_PATH]; static const CHAR testW_txt[] = "\\testW.txt"; trace("SHCreateStreamOnFileW: testing mode %d, STGM flags %08x\n", mode, stgm); /* Don't used a fixed path for the testW.txt file */ GetTempPathA(MAX_PATH, test_fileA); lstrcatA(test_fileA, testW_txt); MultiByteToWideChar(CP_ACP, 0, test_fileA, -1, test_file, MAX_PATH); /* invalid arguments */ if (0) { /* Crashes on NT4 */ stream = NULL; ret = SHCreateStreamOnFileW(NULL, mode | stgm, &stream); ok(ret == HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND) || /* XP */ ret == E_INVALIDARG /* Vista */, "SHCreateStreamOnFileW: expected HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND) or E_INVALIDARG, got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileW: expected a NULL IStream object, got %p\n", stream); } if (0) { /* This test crashes on WinXP SP2 */ ret = SHCreateStreamOnFileW(test_file, mode | stgm, NULL); ok(ret == E_INVALIDARG, "SHCreateStreamOnFileW: expected E_INVALIDARG, got 0x%08x\n", ret); } stream = NULL; ret = SHCreateStreamOnFileW(test_file, mode | STGM_CONVERT | stgm, &stream); ok(ret == E_INVALIDARG, "SHCreateStreamOnFileW: expected E_INVALIDARG, got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileW: expected a NULL IStream object, got %p\n", stream); stream = NULL; ret = SHCreateStreamOnFileW(test_file, mode | STGM_DELETEONRELEASE | stgm, &stream); ok(ret == E_INVALIDARG, "SHCreateStreamOnFileW: expected E_INVALIDARG, got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileW: expected a NULL IStream object, got %p\n", stream); stream = NULL; ret = SHCreateStreamOnFileW(test_file, mode | STGM_TRANSACTED | stgm, &stream); ok(ret == E_INVALIDARG, "SHCreateStreamOnFileW: expected E_INVALIDARG, got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileW: expected a NULL IStream object, got %p\n", stream); /* file does not exist */ stream = NULL; ret = SHCreateStreamOnFileW(test_file, mode | STGM_FAILIFTHERE | stgm, &stream); if (ret == E_INVALIDARG) /* Win98 SE */ { win_skip("Not supported\n"); return; } ok(ret == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND), "SHCreateStreamOnFileW: expected HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND), got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileW: expected a NULL IStream object, got %p\n", stream); stream = NULL; ret = SHCreateStreamOnFileW(test_file, mode | STGM_CREATE | stgm, &stream); ok(ret == S_OK, "SHCreateStreamOnFileW: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileW: expected a valid IStream object, got NULL\n"); if (stream) { test_stream_qi(stream); test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileW: expected 0, got %d\n", refcount); } /* NOTE: don't delete the file, as it will be used for the file exists tests. */ /* file exists */ stream = NULL; ret = SHCreateStreamOnFileW(test_file, mode | STGM_FAILIFTHERE | stgm, &stream); ok(ret == S_OK, "SHCreateStreamOnFileW: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileW: expected a valid IStream object, got NULL\n"); if (stream) { test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileW: expected 0, got %d\n", refcount); } stream = NULL; ret = SHCreateStreamOnFileW(test_file, mode | STGM_CREATE | stgm, &stream); ok(ret == S_OK, "SHCreateStreamOnFileW: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileW: expected a valid IStream object, got NULL\n"); if (stream) { BOOL delret; test_stream_read_write(stream, mode); test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileW: expected 0, got %d\n", refcount); delret = DeleteFileA(test_fileA); ok(delret, "SHCreateStreamOnFileW: could not delete the test file, got error %d\n", GetLastError()); } } static void test_SHCreateStreamOnFileEx(DWORD mode, DWORD stgm) { IStream * stream; IStream * template = NULL; HRESULT ret; ULONG refcount; WCHAR test_file[MAX_PATH]; CHAR test_fileA[MAX_PATH]; static const CHAR testEx_txt[] = "\\testEx.txt"; BOOL delret; if (winetest_debug > 1) trace("SHCreateStreamOnFileEx: testing mode %d, STGM flags %08x\n", mode, stgm); /* Don't used a fixed path for the testEx.txt file */ GetTempPathA(MAX_PATH, test_fileA); lstrcatA(test_fileA, testEx_txt); MultiByteToWideChar(CP_ACP, 0, test_fileA, -1, test_file, MAX_PATH); /* invalid arguments */ if (0) { /* Crashes on NT4 */ stream = NULL; ret = SHCreateStreamOnFileEx(NULL, mode, 0, FALSE, NULL, &stream); ok(ret == HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND) || /* XP */ ret == E_INVALIDARG /* Vista */, "SHCreateStreamOnFileEx: expected HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND) or E_INVALIDARG, got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileEx: expected a NULL IStream object, got %p\n", stream); } stream = NULL; ret = SHCreateStreamOnFileEx(test_file, mode, 0, FALSE, template, &stream); if (ret == HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED)) { win_skip("File probably locked by Anti-Virus/Spam software, trying again\n"); Sleep(1000); ret = SHCreateStreamOnFileEx(test_file, mode, 0, FALSE, template, &stream); } ok( ret == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) || ret == HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER), "SHCreateStreamOnFileEx: expected HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) or " "HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER), got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileEx: expected a NULL IStream object, got %p\n", stream); if (0) { /* This test crashes on WinXP SP2 */ ret = SHCreateStreamOnFileEx(test_file, mode, 0, FALSE, NULL, NULL); ok(ret == E_INVALIDARG, "SHCreateStreamOnFileEx: expected E_INVALIDARG, got 0x%08x\n", ret); } /* file does not exist */ stream = NULL; ret = SHCreateStreamOnFileEx(test_file, mode | STGM_FAILIFTHERE | stgm, 0, FALSE, NULL, &stream); if ((stgm & STGM_TRANSACTED) == STGM_TRANSACTED && mode == STGM_READ) { ok(ret == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) /* XP */ || ret == E_INVALIDARG /* Vista */, "SHCreateStreamOnFileEx: expected E_INVALIDARG or HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND), got 0x%08x\n", ret); if (ret == E_INVALIDARG) { skip("SHCreateStreamOnFileEx: STGM_TRANSACTED not supported in this configuration.\n"); return; } } else { ok( ret == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) || ret == HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER), "SHCreateStreamOnFileEx: expected HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) or " "HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER), got 0x%08x\n", ret); } ok(stream == NULL, "SHCreateStreamOnFileEx: expected a NULL IStream object, got %p\n", stream); stream = NULL; ret = SHCreateStreamOnFileEx(test_file, mode | STGM_FAILIFTHERE | stgm, 0, TRUE, NULL, &stream); /* not supported on win9x */ if (broken(ret == HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER) && stream == NULL)) { skip("Not supported\n"); DeleteFileA(test_fileA); return; } ok(ret == S_OK, "SHCreateStreamOnFileEx: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileEx: expected a valid IStream object, got NULL\n"); if (stream) { test_stream_qi(stream); test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileEx: expected 0, got %d\n", refcount); delret = DeleteFileA(test_fileA); ok(delret, "SHCreateStreamOnFileEx: could not delete the test file, got error %d\n", GetLastError()); } stream = NULL; ret = SHCreateStreamOnFileEx(test_file, mode | STGM_CREATE | stgm, 0, FALSE, NULL, &stream); if (ret == HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED)) { win_skip("File probably locked by Anti-Virus/Spam software, trying again\n"); Sleep(1000); ret = SHCreateStreamOnFileEx(test_file, mode | STGM_CREATE | stgm, 0, FALSE, NULL, &stream); } ok(ret == S_OK, "SHCreateStreamOnFileEx: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileEx: expected a valid IStream object, got NULL\n"); if (stream) { test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileEx: expected 0, got %d\n", refcount); delret = DeleteFileA(test_fileA); ok(delret, "SHCreateStreamOnFileEx: could not delete the test file, got error %d\n", GetLastError()); } stream = NULL; ret = SHCreateStreamOnFileEx(test_file, mode | STGM_CREATE | stgm, 0, TRUE, NULL, &stream); if (ret == HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED)) { win_skip("File probably locked by Anti-Virus/Spam software, trying again\n"); Sleep(1000); ret = SHCreateStreamOnFileEx(test_file, mode | STGM_CREATE | stgm, 0, TRUE, NULL, &stream); } ok(ret == S_OK, "SHCreateStreamOnFileEx: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileEx: expected a valid IStream object, got NULL\n"); if (stream) { test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileEx: expected 0, got %d\n", refcount); } /* NOTE: don't delete the file, as it will be used for the file exists tests. */ /* file exists */ stream = NULL; ret = SHCreateStreamOnFileEx(test_file, mode | STGM_FAILIFTHERE | stgm, 0, FALSE, NULL, &stream); ok(ret == S_OK, "SHCreateStreamOnFileEx: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileEx: expected a valid IStream object, got NULL\n"); if (stream) { test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileEx: expected 0, got %d\n", refcount); } stream = NULL; ret = SHCreateStreamOnFileEx(test_file, mode | STGM_FAILIFTHERE | stgm, 0, TRUE, NULL, &stream); ok(ret == HRESULT_FROM_WIN32(ERROR_FILE_EXISTS), "SHCreateStreamOnFileEx: expected HRESULT_FROM_WIN32(ERROR_FILE_EXISTS), got 0x%08x\n", ret); ok(stream == NULL, "SHCreateStreamOnFileEx: expected a NULL IStream object, got %p\n", stream); stream = NULL; ret = SHCreateStreamOnFileEx(test_file, mode | STGM_CREATE | stgm, 0, FALSE, NULL, &stream); ok(ret == S_OK, "SHCreateStreamOnFileEx: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileEx: expected a valid IStream object, got NULL\n"); if (stream) { test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileEx: expected 0, got %d\n", refcount); } stream = NULL; ret = SHCreateStreamOnFileEx(test_file, mode | STGM_CREATE | stgm, 0, TRUE, NULL, &stream); ok(ret == S_OK, "SHCreateStreamOnFileEx: expected S_OK, got 0x%08x\n", ret); ok(stream != NULL, "SHCreateStreamOnFileEx: expected a valid IStream object, got NULL\n"); if (stream) { test_IStream_invalid_operations(stream, mode); refcount = IStream_Release(stream); ok(refcount == 0, "SHCreateStreamOnFileEx: expected 0, got %d\n", refcount); } delret = DeleteFileA(test_fileA); ok(delret, "SHCreateStreamOnFileEx: could not delete the test file, got error %d\n", GetLastError()); } static void test_SHCreateStreamOnFileEx_CopyTo(void) { HRESULT ret; IStream *src, *dst; WCHAR tmpPath[MAX_PATH]; WCHAR srcFileName[MAX_PATH]; WCHAR dstFileName[MAX_PATH]; ULARGE_INTEGER count, read, written; LARGE_INTEGER distance; static const char srcContents[1]; static const WCHAR prefix[] = { 'T', 'S', 'T', 0 }; GetTempPathW(MAX_PATH, tmpPath); ret = GetTempFileNameW(tmpPath, prefix, 0, srcFileName); ok(ret != 0, "GetTempFileName failed, got error %d\n", GetLastError()); ret = GetTempFileNameW(tmpPath, prefix, 0, dstFileName); ok(ret != 0, "GetTempFileName failed, got error %d\n", GetLastError()); ret = SHCreateStreamOnFileEx(srcFileName, STGM_CREATE | STGM_READWRITE | STGM_DELETEONRELEASE, FILE_ATTRIBUTE_TEMPORARY, FALSE, NULL, &src); ok(SUCCEEDED(ret), "SHCreateStreamOnFileEx failed with ret=0x%08x\n", ret); written.QuadPart = 0; ret = IStream_Write(src, srcContents, sizeof(srcContents), &U(written).LowPart); ok(SUCCEEDED(ret), "ISequentialStream_Write failed with ret=0x%08x\n", ret); distance.QuadPart = 0; ret = IStream_Seek(src, distance, STREAM_SEEK_SET, &written); ok(SUCCEEDED(ret), "ISequentialStream_Seek failed with ret=0x%08x\n", ret); ret = SHCreateStreamOnFileEx(dstFileName, STGM_CREATE | STGM_READWRITE | STGM_DELETEONRELEASE, FILE_ATTRIBUTE_TEMPORARY, FALSE, NULL, &dst); ok(SUCCEEDED(ret), "SHCreateStreamOnFileEx failed with ret=0x%08x\n", ret); /* Test using a count larger than the source file, so that the Read operation will fall short */ count.QuadPart = 2; ret = IStream_CopyTo(src, dst, count, &read, &written); ok(SUCCEEDED(ret), "CopyTo failed with ret=0x%08x\n", ret); ok(read.QuadPart == 1, "read does not match size: %d != 1\n", U(read).LowPart); ok(written.QuadPart == 1, "written does not match size: %d != 1\n", U(written).LowPart); IStream_Release(dst); IStream_Release(src); DeleteFileW( srcFileName ); DeleteFileW( dstFileName ); } static void test_SHCreateMemStream(void) { static const BYTE initial[10]; IStream *stream, *stream2; LARGE_INTEGER off; IUnknown *unk; char buff[10]; ULONG i, num; HRESULT hr; stream = SHCreateMemStream(initial, 0); ok(stream != NULL, "Failed to create a stream.\n"); IStream_Release(stream); stream = SHCreateMemStream(NULL, 10); ok(stream != NULL, "Failed to create a stream.\n"); IStream_Release(stream); stream = SHCreateMemStream(NULL, 0); ok(stream != NULL, "Failed to create a stream.\n"); hr = IStream_QueryInterface(stream, &IID_ISequentialStream, (void **)&unk); ok(hr == S_OK || broken(hr == E_NOINTERFACE) /* WinXP */, "Failed to QI, hr %#x.\n", hr); if (unk) IUnknown_Release(unk); num = ~0; hr = IStream_Read(stream, buff, sizeof(buff), &num); ok(hr == S_FALSE, "Unexpected hr %#x.\n", hr); ok(num == 0, "expected 0, got 0x%08x\n", num); num = ~0; memset(buff, 0x55, sizeof(buff)); hr = IStream_Write(stream, buff, sizeof(buff), &num); ok(hr == S_OK, "Failed to write, hr %#x.\n", hr); ok(num == sizeof(buff), "expected %u, got %u\n", sizeof(buff), num); off.QuadPart = 0; hr = IStream_Seek(stream, off, STREAM_SEEK_SET, NULL); ok(hr == S_OK, "Failed to seek, hr %#x.\n", hr); num = ~0; memset(buff, 0, sizeof(buff)); hr = IStream_Read(stream, buff, sizeof(buff), &num); ok(hr == S_OK, "Failed to read, hr %#x.\n", hr); ok(num == sizeof(buff), "expected %u, got %u\n", sizeof(buff), num); for (i = 0; i < ARRAY_SIZE(buff); i++) ok(buff[i] == 0x55, "expected 0x55, got 0x%02x at %u\n", buff[i], i); num = ~0; hr = IStream_Read(stream, buff, sizeof(buff), &num); ok(hr == S_FALSE, "Unexpected hr %#x.\n", hr); ok(num == 0, "expected 0, got 0x%08x\n", num); num = ~0; memset(buff, 0x11, sizeof(buff)); hr = IStream_Write(stream, buff, sizeof(buff), &num); ok(hr == S_OK, "Failed to write, hr %#x.\n", hr); ok(num == sizeof(buff), "expected %u, got %u\n", sizeof(buff), num); off.QuadPart = -sizeof(buff); hr = IStream_Seek(stream, off, STREAM_SEEK_CUR, NULL); ok(hr == S_OK, "Failed to seek, hr %#x.\n", hr); num = ~0; memset(buff, 0, sizeof(buff)); hr = IStream_Read(stream, buff, sizeof(buff), &num); ok(hr == S_OK, "Failed to read, hr %#x.\n", hr); ok(num == sizeof(buff), "expected %u, got %u\n", sizeof(buff), num); for (i = 0; i < ARRAY_SIZE(buff); i++) ok(buff[i] == 0x11, "expected 0x11, got 0x%02x at %u\n", buff[i], i); num = ~0; hr = IStream_Read(stream, buff, sizeof(buff), &num); ok(hr == S_FALSE, "Unexpected hr %#x.\n", hr); ok(num == 0, "expected 0, got 0x%08x\n", num); hr = IStream_Clone(stream, &stream2); todo_wine ok(hr == S_OK || broken(hr == E_NOTIMPL) /* < Win8 */, "Failed to clone a stream, hr %#x.\n", hr); if (hr == S_OK) IStream_Release(stream2); IStream_Release(stream); } START_TEST(istream) { static const DWORD stgm_access[] = { STGM_READ, STGM_WRITE, STGM_READWRITE }; static const DWORD stgm_sharing[] = { 0, STGM_SHARE_DENY_NONE, STGM_SHARE_DENY_READ, STGM_SHARE_DENY_WRITE, STGM_SHARE_EXCLUSIVE }; static const DWORD stgm_flags[] = { 0, STGM_CONVERT, STGM_DELETEONRELEASE, STGM_CONVERT | STGM_DELETEONRELEASE, STGM_TRANSACTED | STGM_CONVERT, STGM_TRANSACTED | STGM_DELETEONRELEASE, STGM_TRANSACTED | STGM_CONVERT | STGM_DELETEONRELEASE }; int i, j, k; for (i = 0; i != ARRAY_SIZE(stgm_access); i++) { for (j = 0; j != ARRAY_SIZE(stgm_sharing); j ++) { test_SHCreateStreamOnFileA(stgm_access[i], stgm_sharing[j]); test_SHCreateStreamOnFileW(stgm_access[i], stgm_sharing[j]); for (k = 0; k != ARRAY_SIZE(stgm_flags); k++) test_SHCreateStreamOnFileEx(stgm_access[i], stgm_sharing[j] | stgm_flags[k]); } } test_SHCreateStreamOnFileEx_CopyTo(); test_SHCreateMemStream(); }