Wine-Bug: https://bugs.winehq.org/show_bug.cgi?id=42710 Signed-off-by: Jactry Zeng jzeng@codeweavers.com --- dlls/ole32/ole2impl.c | 51 ++++++++++++++++++++++++++++++++++-- dlls/ole32/tests/ole2.c | 57 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 106 insertions(+), 2 deletions(-)
diff --git a/dlls/ole32/ole2impl.c b/dlls/ole32/ole2impl.c index fa5777beb0..73d8210101 100644 --- a/dlls/ole32/ole2impl.c +++ b/dlls/ole32/ole2impl.c @@ -232,9 +232,56 @@ HRESULT WINAPI OleCreateStaticFromData(IDataObject *data, REFIID iid, IOleClientSite *client_site, IStorage *stg, void **obj) { - FIXME("%p,%s,%08x,%p,%p,%p,%p: semi-stub\n", + HRESULT hr; + IOleCache2 *ole_cache = NULL; + IPersistStorage *persist = NULL; + DWORD connection; + STGMEDIUM stgmedium; + + TRACE("(%p, %s, 0x%08x, %p, %p, %p, %p)\n", data, debugstr_guid(iid), renderopt, fmt, client_site, stg, obj); - return OleCreateFromData(data, iid, renderopt, fmt, client_site, stg, obj); + + if (!obj || !stg) + return E_INVALIDARG; + + if (renderopt != OLERENDER_FORMAT) + { + FIXME("semi-stub\n"); + return OleCreateFromData(data, iid, renderopt, fmt, client_site, stg, obj); + } + + if (!fmt) + return E_INVALIDARG; + + hr = OleCreateDefaultHandler(&CLSID_NULL, NULL, iid, (void **)obj); + if (FAILED(hr)) goto end; + + hr = IUnknown_QueryInterface((IUnknown *)*obj, &IID_IOleCache2, (void **)&ole_cache); + if (FAILED(hr)) goto end; + + hr = IOleCache2_Cache(ole_cache, fmt, ADVF_PRIMEFIRST, &connection); + if (FAILED(hr)) goto end; + + hr = IDataObject_GetData(data, fmt, &stgmedium); + if (FAILED(hr)) goto end; + + hr = IOleCache2_SetData(ole_cache, fmt, &stgmedium, TRUE); + if (FAILED(hr)) goto end; + + hr = IOleCache2_QueryInterface(ole_cache, &IID_IPersistStorage, (void **)&persist); + if (FAILED(hr)) goto end; + + hr = IPersistStorage_Save(persist, stg, TRUE); + if (FAILED(hr)) goto end; + + hr = IPersistStorage_SaveCompleted(persist, NULL); + +end: + if (persist) + IPersistStorage_Release(persist); + if (ole_cache) + IOleCache2_Release(ole_cache); + return hr; }
/****************************************************************************** diff --git a/dlls/ole32/tests/ole2.c b/dlls/ole32/tests/ole2.c index 48c9e0cbab..ceec369dc3 100644 --- a/dlls/ole32/tests/ole2.c +++ b/dlls/ole32/tests/ole2.c @@ -4593,6 +4593,62 @@ todo_wine_if(!(test_data[i].in == &stg_def_0 || test_data[i].in == &stg_def_4 || } }
+static void test_OleCreateStaticFromData(void) +{ + HRESULT hr; + IOleObject *ole_obj = NULL; + IStorage *storage; + ILockBytes *ilb; + static FORMATETC bitmap_fmt[] = + { + { CF_BITMAP, NULL, DVASPECT_CONTENT, -1, TYMED_GDI }, + { 0 } + }; + static const struct expected_method methods_create_from_bitmap[] = + { + { "DataObject_EnumFormatEtc", TEST_TODO }, + { "DataObject_GetDataHere", 0 }, + { "DataObject_QueryGetData", 0, { CF_METAFILEPICT, NULL, DVASPECT_CONTENT, -1, TYMED_ISTORAGE } }, + { NULL } + }; + static const struct expected_method methods_createstatic_from_bitmap[] = + { + { "DataObject_GetData", 0, { CF_BITMAP, NULL, DVASPECT_CONTENT, -1, TYMED_GDI } }, + { NULL } + }; + + hr = CreateILockBytesOnHGlobal(NULL, TRUE, &ilb); + ok(hr == S_OK, "CreateILockBytesOnHGlobal failed: 0x%08x.\n", hr); + hr = StgCreateDocfileOnILockBytes(ilb, STGM_SHARE_EXCLUSIVE | STGM_CREATE | STGM_READWRITE, + 0, &storage); + ok(hr == S_OK, "StgCreateDocfileOnILockBytes failed: 0x%08x.\n", hr); + + hr = OleCreateStaticFromData(&DataObject, &IID_IOleObject, OLERENDER_FORMAT, + bitmap_fmt, NULL, NULL, (void **)&ole_obj); + ok(hr == E_INVALIDARG, "OleCreateStaticFromData should fail: 0x%08x.\n", hr); + + hr = OleCreateStaticFromData(&DataObject, &IID_IOleObject, OLERENDER_FORMAT, + bitmap_fmt, NULL, storage, NULL); + ok(hr == E_INVALIDARG, "OleCreateStaticFromData should fail: 0x%08x.\n", hr); + + g_dataobject_fmts = bitmap_fmt; + expected_method_list = methods_create_from_bitmap; + hr = OleCreateFromData(&DataObject, &IID_IOleObject, OLERENDER_FORMAT, bitmap_fmt, NULL, + storage, (void **)&ole_obj); + todo_wine ok(hr == DV_E_FORMATETC, "OleCreateFromData should failed: 0x%08x.\n", hr); + + g_dataobject_fmts = bitmap_fmt; + expected_method_list = methods_createstatic_from_bitmap; + hr = OleCreateStaticFromData(&DataObject, &IID_IOleObject, OLERENDER_FORMAT, + bitmap_fmt, NULL, storage, (void **)&ole_obj); + ok(hr == S_OK, "OleCreateStaticFromData failed: 0x%08x.\n", hr); + if (ole_obj) + IOleObject_Release(ole_obj); + + IStorage_Release(storage); + ILockBytes_Release(ilb); +} + START_TEST(ole2) { DWORD dwRegister; @@ -4643,6 +4699,7 @@ START_TEST(ole2) test_data_cache_save(); test_data_cache_save_data(); test_data_cache_contents(); + test_OleCreateStaticFromData();
CoUninitialize(); }
On Wed, Aug 29, 2018 at 04:08:57PM +0800, Jactry Zeng wrote:
Wine-Bug: https://bugs.winehq.org/show_bug.cgi?id=42710 Signed-off-by: Jactry Zeng jzeng@codeweavers.com
dlls/ole32/ole2impl.c | 51 ++++++++++++++++++++++++++++++++++-- dlls/ole32/tests/ole2.c | 57 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 106 insertions(+), 2 deletions(-)
diff --git a/dlls/ole32/ole2impl.c b/dlls/ole32/ole2impl.c index fa5777beb0..73d8210101 100644 --- a/dlls/ole32/ole2impl.c +++ b/dlls/ole32/ole2impl.c @@ -232,9 +232,56 @@ HRESULT WINAPI OleCreateStaticFromData(IDataObject *data, REFIID iid, IOleClientSite *client_site, IStorage *stg, void **obj) {
- FIXME("%p,%s,%08x,%p,%p,%p,%p: semi-stub\n",
- HRESULT hr;
- IOleCache2 *ole_cache = NULL;
- IPersistStorage *persist = NULL;
- DWORD connection;
- STGMEDIUM stgmedium;
- TRACE("(%p, %s, 0x%08x, %p, %p, %p, %p)\n", data, debugstr_guid(iid), renderopt, fmt, client_site, stg, obj);
- return OleCreateFromData(data, iid, renderopt, fmt, client_site, stg, obj);
- if (!obj || !stg)
return E_INVALIDARG;
- if (renderopt != OLERENDER_FORMAT)
- {
FIXME("semi-stub\n");
return OleCreateFromData(data, iid, renderopt, fmt, client_site, stg, obj);
- }
- if (!fmt)
return E_INVALIDARG;
- hr = OleCreateDefaultHandler(&CLSID_NULL, NULL, iid, (void **)obj);
You should call this with the correct CLSID. A switch on fmt->cFormat with CF_DIB/CF_BITMAP -> CLSID_PictureDib and the two metafile formats going to their respective CLSID_Picture* classes should do it.
Also, retrieve the IID_IOleObject interface directly and then QI on iid near the end of the function.
You'll need to also call IOleObject_SetClientSite() if client_site is non-NULL.
- if (FAILED(hr)) goto end;
- hr = IUnknown_QueryInterface((IUnknown *)*obj, &IID_IOleCache2, (void **)&ole_cache);
This is now IOleObject_QI
- if (FAILED(hr)) goto end;
- hr = IOleCache2_Cache(ole_cache, fmt, ADVF_PRIMEFIRST, &connection);
- if (FAILED(hr)) goto end;
- hr = IDataObject_GetData(data, fmt, &stgmedium);
- if (FAILED(hr)) goto end;
I would do the GetData call before creating the default handler, that way you can bail without the cost of creating the default handler if GetData fails. You need to ReleaseStgMedium() if there's a failure before a successful call to IOleCache2_SetData.
- hr = IOleCache2_SetData(ole_cache, fmt, &stgmedium, TRUE);
- if (FAILED(hr)) goto end;
- hr = IOleCache2_QueryInterface(ole_cache, &IID_IPersistStorage, (void **)&persist);
- if (FAILED(hr)) goto end;
- hr = IPersistStorage_Save(persist, stg, TRUE);
- if (FAILED(hr)) goto end;
Does the storage actually get saved at this point? I would imagine you'd do a WriteClassStg() and a IPersistStorage_InitNew() here, but some tests investigating the state of the storage after this function returns would be good.
- hr = IPersistStorage_SaveCompleted(persist, NULL);
+end:
- if (persist)
IPersistStorage_Release(persist);
- if (ole_cache)
IOleCache2_Release(ole_cache);
- return hr;
}
On Wed, Aug 29, 2018 at 10:08:16AM +0100, Huw Davies wrote:
On Wed, Aug 29, 2018 at 04:08:57PM +0800, Jactry Zeng wrote:
Wine-Bug: https://bugs.winehq.org/show_bug.cgi?id=42710 Signed-off-by: Jactry Zeng jzeng@codeweavers.com
dlls/ole32/ole2impl.c | 51 ++++++++++++++++++++++++++++++++++-- dlls/ole32/tests/ole2.c | 57 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 106 insertions(+), 2 deletions(-)
diff --git a/dlls/ole32/ole2impl.c b/dlls/ole32/ole2impl.c index fa5777beb0..73d8210101 100644 --- a/dlls/ole32/ole2impl.c +++ b/dlls/ole32/ole2impl.c @@ -232,9 +232,56 @@ HRESULT WINAPI OleCreateStaticFromData(IDataObject *data, REFIID iid, IOleClientSite *client_site, IStorage *stg, void **obj) {
- FIXME("%p,%s,%08x,%p,%p,%p,%p: semi-stub\n",
- HRESULT hr;
- IOleCache2 *ole_cache = NULL;
- IPersistStorage *persist = NULL;
- DWORD connection;
- STGMEDIUM stgmedium;
- TRACE("(%p, %s, 0x%08x, %p, %p, %p, %p)\n", data, debugstr_guid(iid), renderopt, fmt, client_site, stg, obj);
- return OleCreateFromData(data, iid, renderopt, fmt, client_site, stg, obj);
- if (!obj || !stg)
return E_INVALIDARG;
- if (renderopt != OLERENDER_FORMAT)
- {
FIXME("semi-stub\n");
return OleCreateFromData(data, iid, renderopt, fmt, client_site, stg, obj);
- }
- if (!fmt)
return E_INVALIDARG;
- hr = OleCreateDefaultHandler(&CLSID_NULL, NULL, iid, (void **)obj);
You should call this with the correct CLSID. A switch on fmt->cFormat with CF_DIB/CF_BITMAP -> CLSID_PictureDib and the two metafile formats going to their respective CLSID_Picture* classes should do it.
Also, retrieve the IID_IOleObject interface directly and then QI on iid near the end of the function.
You'll need to also call IOleObject_SetClientSite() if client_site is non-NULL.
- if (FAILED(hr)) goto end;
- hr = IUnknown_QueryInterface((IUnknown *)*obj, &IID_IOleCache2, (void **)&ole_cache);
This is now IOleObject_QI
- if (FAILED(hr)) goto end;
- hr = IOleCache2_Cache(ole_cache, fmt, ADVF_PRIMEFIRST, &connection);
- if (FAILED(hr)) goto end;
- hr = IDataObject_GetData(data, fmt, &stgmedium);
- if (FAILED(hr)) goto end;
I would do the GetData call before creating the default handler, that way you can bail without the cost of creating the default handler if GetData fails. You need to ReleaseStgMedium() if there's a failure before a successful call to IOleCache2_SetData.
- hr = IOleCache2_SetData(ole_cache, fmt, &stgmedium, TRUE);
- if (FAILED(hr)) goto end;
- hr = IOleCache2_QueryInterface(ole_cache, &IID_IPersistStorage, (void **)&persist);
- if (FAILED(hr)) goto end;
- hr = IPersistStorage_Save(persist, stg, TRUE);
- if (FAILED(hr)) goto end;
Does the storage actually get saved at this point? I would imagine you'd do a WriteClassStg() and a IPersistStorage_InitNew() here, but some tests investigating the state of the storage after this function returns would be good.
Just to be clear, you'd do this before the calls to IOleCache2_Cache and _SetData.
Huw.