Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-cw
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
wine
wine-cw
Commits
d03c1cfc
Commit
d03c1cfc
authored
Jun 17, 2014
by
Andrew Eikum
Committed by
Alexandre Julliard
Jun 18, 2014
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
packager: Add loading support for Ole10Native storage objects.
parent
b456eb52
Show whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
684 additions
and
2 deletions
+684
-2
configure
configure
+1
-0
configure.ac
configure.ac
+1
-0
Makefile.in
dlls/packager/Makefile.in
+1
-1
packager_main.c
dlls/packager/packager_main.c
+276
-1
Makefile.in
dlls/packager/tests/Makefile.in
+5
-0
oleobj.c
dlls/packager/tests/oleobj.c
+400
-0
No files found.
configure
View file @
d03c1cfc
...
...
@@ -17102,6 +17102,7 @@ wine_fn_config_dll opencl enable_opencl
wine_fn_config_dll opengl32 enable_opengl32 implib
wine_fn_config_test dlls/opengl32/tests opengl32_test
wine_fn_config_dll packager enable_packager clean
wine_fn_config_test dlls/packager/tests packager_test
wine_fn_config_dll pdh enable_pdh implib
wine_fn_config_test dlls/pdh/tests pdh_test
wine_fn_config_dll photometadatahandler enable_photometadatahandler
...
...
configure.ac
View file @
d03c1cfc
...
...
@@ -3076,6 +3076,7 @@ WINE_CONFIG_DLL(opencl)
WINE_CONFIG_DLL(opengl32,,[implib])
WINE_CONFIG_TEST(dlls/opengl32/tests)
WINE_CONFIG_DLL(packager,,[clean])
WINE_CONFIG_TEST(dlls/packager/tests)
WINE_CONFIG_DLL(pdh,,[implib])
WINE_CONFIG_TEST(dlls/pdh/tests)
WINE_CONFIG_DLL(photometadatahandler)
...
...
dlls/packager/Makefile.in
View file @
d03c1cfc
MODULE
=
packager.dll
IMPORTS
=
uuid
IMPORTS
=
uuid
shell32 shlwapi user32
C_SRCS
=
\
packager_main.c
...
...
dlls/packager/packager_main.c
View file @
d03c1cfc
...
...
@@ -26,6 +26,9 @@
#include "winbase.h"
#include "ole2.h"
#include "rpcproxy.h"
#include "shellapi.h"
#include "shlwapi.h"
#include "wine/debug.h"
#include "packager_classes.h"
...
...
@@ -36,8 +39,11 @@ static HINSTANCE g_instance;
struct
Package
{
IOleObject
IOleObject_iface
;
IPersistStorage
IPersistStorage_iface
;
LONG
ref
;
WCHAR
filename
[
MAX_PATH
];
};
static
inline
struct
Package
*
impl_from_IOleObject
(
IOleObject
*
iface
)
...
...
@@ -45,6 +51,11 @@ static inline struct Package *impl_from_IOleObject(IOleObject *iface)
return
CONTAINING_RECORD
(
iface
,
struct
Package
,
IOleObject_iface
);
}
static
inline
struct
Package
*
impl_from_IPersistStorage
(
IPersistStorage
*
iface
)
{
return
CONTAINING_RECORD
(
iface
,
struct
Package
,
IPersistStorage_iface
);
}
static
HRESULT
WINAPI
OleObject_QueryInterface
(
IOleObject
*
iface
,
REFIID
riid
,
void
**
obj
)
{
struct
Package
*
This
=
impl_from_IOleObject
(
iface
);
...
...
@@ -53,6 +64,9 @@ static HRESULT WINAPI OleObject_QueryInterface(IOleObject *iface, REFIID riid, v
IsEqualGUID
(
riid
,
&
IID_IOleObject
))
{
TRACE
(
"(%p)->(IID_IOleObject, %p)
\n
"
,
This
,
obj
);
*
obj
=
&
This
->
IOleObject_iface
;
}
else
if
(
IsEqualGUID
(
riid
,
&
IID_IPersistStorage
)){
TRACE
(
"(%p)->(IID_IPersistStorage, %p)
\n
"
,
This
,
obj
);
*
obj
=
&
This
->
IPersistStorage_iface
;
}
else
{
FIXME
(
"(%p)->(%s, %p)
\n
"
,
This
,
debugstr_guid
(
riid
),
obj
);
*
obj
=
NULL
;
...
...
@@ -80,8 +94,12 @@ static ULONG WINAPI OleObject_Release(IOleObject *iface)
TRACE
(
"(%p) ref=%d
\n
"
,
This
,
ref
);
if
(
!
ref
)
if
(
!
ref
){
if
(
*
This
->
filename
)
DeleteFileW
(
This
->
filename
);
HeapFree
(
GetProcessHeap
(),
0
,
This
);
}
return
ref
;
}
...
...
@@ -262,6 +280,262 @@ static const IOleObjectVtbl OleObject_Vtbl = {
OleObject_SetColorScheme
};
static
HRESULT
WINAPI
PersistStorage_QueryInterface
(
IPersistStorage
*
iface
,
REFIID
riid
,
void
**
ppvObject
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
return
OleObject_QueryInterface
(
&
This
->
IOleObject_iface
,
riid
,
ppvObject
);
}
static
ULONG
WINAPI
PersistStorage_AddRef
(
IPersistStorage
*
iface
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
return
OleObject_AddRef
(
&
This
->
IOleObject_iface
);
}
static
ULONG
WINAPI
PersistStorage_Release
(
IPersistStorage
*
iface
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
return
OleObject_Release
(
&
This
->
IOleObject_iface
);
}
static
HRESULT
WINAPI
PersistStorage_GetClassID
(
IPersistStorage
*
iface
,
CLSID
*
pClassID
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
FIXME
(
"(%p)->(%p)
\n
"
,
This
,
pClassID
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
PersistStorage_IsDirty
(
IPersistStorage
*
iface
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
FIXME
(
"(%p)
\n
"
,
This
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
PersistStorage_InitNew
(
IPersistStorage
*
iface
,
IStorage
*
pStg
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
FIXME
(
"(%p)->(%p)
\n
"
,
This
,
pStg
);
return
E_NOTIMPL
;
}
static
HRESULT
discard_string
(
struct
Package
*
This
,
IStream
*
stream
)
{
ULONG
nbytes
;
HRESULT
hr
;
char
chr
=
0
;
do
{
hr
=
IStream_Read
(
stream
,
&
chr
,
1
,
&
nbytes
);
if
(
FAILED
(
hr
)
||
!
nbytes
){
TRACE
(
"Unexpected end of stream or Read failed with %08x
\n
"
,
hr
);
return
(
hr
==
S_OK
||
hr
==
S_FALSE
)
?
E_FAIL
:
hr
;
}
}
while
(
chr
);
return
S_OK
;
}
static
HRESULT
WINAPI
PersistStorage_Load
(
IPersistStorage
*
iface
,
IStorage
*
pStg
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
IStream
*
stream
;
DWORD
payload_size
,
len
,
stream_filename_len
,
filenameA_len
,
i
,
bytes_read
;
ULARGE_INTEGER
payload_pos
;
LARGE_INTEGER
seek
;
HRESULT
hr
;
HANDLE
file
=
INVALID_HANDLE_VALUE
;
WCHAR
filenameW
[
MAX_PATH
];
char
filenameA
[
MAX_PATH
];
WCHAR
*
stream_filename
;
WCHAR
*
base_end
,
extension
[
MAX_PATH
];
static
const
WCHAR
ole10nativeW
[]
=
{
0x0001
,
'O'
,
'l'
,
'e'
,
'1'
,
'0'
,
'N'
,
'a'
,
't'
,
'i'
,
'v'
,
'e'
,
0
};
TRACE
(
"(%p)->(%p)
\n
"
,
This
,
pStg
);
hr
=
IStorage_OpenStream
(
pStg
,
ole10nativeW
,
NULL
,
STGM_READ
|
STGM_SHARE_EXCLUSIVE
,
0
,
&
stream
);
if
(
FAILED
(
hr
)){
TRACE
(
"OpenStream gave: %08x
\n
"
,
hr
);
return
hr
;
}
/* skip stream size & two unknown bytes */
seek
.
QuadPart
=
6
;
hr
=
IStream_Seek
(
stream
,
seek
,
STREAM_SEEK_SET
,
NULL
);
if
(
FAILED
(
hr
))
goto
exit
;
/* read and discard label */
hr
=
discard_string
(
This
,
stream
);
if
(
FAILED
(
hr
))
goto
exit
;
/* read and discard filename */
hr
=
discard_string
(
This
,
stream
);
if
(
FAILED
(
hr
))
goto
exit
;
/* skip more unknown data */
seek
.
QuadPart
=
4
;
hr
=
IStream_Seek
(
stream
,
seek
,
STREAM_SEEK_CUR
,
NULL
);
if
(
FAILED
(
hr
))
goto
exit
;
/* ASCIIZ filename */
hr
=
IStream_Read
(
stream
,
&
filenameA_len
,
4
,
NULL
);
if
(
FAILED
(
hr
))
goto
exit
;
hr
=
IStream_Read
(
stream
,
filenameA
,
filenameA_len
,
NULL
);
if
(
FAILED
(
hr
))
goto
exit
;
/* skip payload for now */
hr
=
IStream_Read
(
stream
,
&
payload_size
,
4
,
NULL
);
if
(
FAILED
(
hr
))
goto
exit
;
seek
.
QuadPart
=
0
;
hr
=
IStream_Seek
(
stream
,
seek
,
STREAM_SEEK_CUR
,
&
payload_pos
);
if
(
FAILED
(
hr
))
goto
exit
;
seek
.
QuadPart
=
payload_size
;
hr
=
IStream_Seek
(
stream
,
seek
,
STREAM_SEEK_CUR
,
NULL
);
if
(
FAILED
(
hr
))
goto
exit
;
/* read WCHAR filename, if present */
hr
=
IStream_Read
(
stream
,
&
len
,
4
,
&
bytes_read
);
if
(
SUCCEEDED
(
hr
)
&&
bytes_read
==
4
&&
len
>
0
){
hr
=
IStream_Read
(
stream
,
filenameW
,
len
*
sizeof
(
WCHAR
),
NULL
);
if
(
FAILED
(
hr
))
goto
exit
;
}
else
{
len
=
MultiByteToWideChar
(
CP_ACP
,
0
,
filenameA
,
filenameA_len
,
filenameW
,
sizeof
(
filenameW
)
/
sizeof
(
*
filenameW
));
}
stream_filename
=
filenameW
+
len
-
1
;
while
(
stream_filename
!=
filenameW
&&
*
stream_filename
!=
'\\'
)
--
stream_filename
;
if
(
*
stream_filename
==
'\\'
)
++
stream_filename
;
stream_filename_len
=
len
-
(
stream_filename
-
filenameW
);
len
=
GetTempPathW
(
sizeof
(
This
->
filename
),
This
->
filename
);
memcpy
(
This
->
filename
+
len
,
stream_filename
,
stream_filename_len
*
sizeof
(
WCHAR
));
This
->
filename
[
len
+
stream_filename_len
]
=
0
;
/* read & write payload */
memcpy
(
&
seek
,
&
payload_pos
,
sizeof
(
seek
));
/* STREAM_SEEK_SET treats as ULARGE_INTEGER */
hr
=
IStream_Seek
(
stream
,
seek
,
STREAM_SEEK_SET
,
NULL
);
if
(
FAILED
(
hr
))
goto
exit
;
base_end
=
PathFindExtensionW
(
This
->
filename
);
lstrcpyW
(
extension
,
base_end
);
i
=
1
;
file
=
CreateFileW
(
This
->
filename
,
GENERIC_WRITE
,
FILE_SHARE_READ
,
NULL
,
CREATE_NEW
,
FILE_ATTRIBUTE_NORMAL
,
NULL
);
while
(
file
==
INVALID_HANDLE_VALUE
){
static
const
WCHAR
fmtW
[]
=
{
' '
,
'('
,
'%'
,
'u'
,
')'
,
0
};
if
(
GetLastError
()
!=
ERROR_FILE_EXISTS
){
WARN
(
"CreateFile failed: %u
\n
"
,
GetLastError
());
hr
=
E_FAIL
;
goto
exit
;
}
/* file exists, so increment file name and try again */
++
i
;
wsprintfW
(
base_end
,
fmtW
,
i
);
lstrcatW
(
base_end
,
extension
);
file
=
CreateFileW
(
This
->
filename
,
GENERIC_WRITE
,
FILE_SHARE_READ
,
NULL
,
CREATE_NEW
,
FILE_ATTRIBUTE_NORMAL
,
NULL
);
}
TRACE
(
"Final filename: %s
\n
"
,
wine_dbgstr_w
(
This
->
filename
));
while
(
payload_size
){
ULONG
nbytes
;
BYTE
data
[
4096
];
DWORD
written
;
hr
=
IStream_Read
(
stream
,
data
,
min
(
sizeof
(
data
),
payload_size
),
&
nbytes
);
if
(
FAILED
(
hr
)
||
nbytes
==
0
){
TRACE
(
"Unexpected end of file, or Read failed with %08x
\n
"
,
hr
);
if
(
hr
==
S_OK
||
hr
==
S_FALSE
)
hr
=
E_FAIL
;
goto
exit
;
}
payload_size
-=
nbytes
;
WriteFile
(
file
,
data
,
nbytes
,
&
written
,
NULL
);
}
hr
=
S_OK
;
exit:
if
(
file
!=
INVALID_HANDLE_VALUE
){
CloseHandle
(
file
);
if
(
FAILED
(
hr
))
DeleteFileW
(
This
->
filename
);
}
IStream_Release
(
stream
);
TRACE
(
"Returning: %08x
\n
"
,
hr
);
return
hr
;
}
static
HRESULT
WINAPI
PersistStorage_Save
(
IPersistStorage
*
iface
,
IStorage
*
pStgSave
,
BOOL
fSameAsLoad
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
FIXME
(
"(%p)->(%p, %u)
\n
"
,
This
,
pStgSave
,
fSameAsLoad
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
PersistStorage_SaveCompleted
(
IPersistStorage
*
iface
,
IStorage
*
pStgNew
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
FIXME
(
"(%p)->(%p)
\n
"
,
This
,
pStgNew
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
PersistStorage_HandsOffStorage
(
IPersistStorage
*
iface
)
{
struct
Package
*
This
=
impl_from_IPersistStorage
(
iface
);
FIXME
(
"(%p)
\n
"
,
This
);
return
E_NOTIMPL
;
}
static
IPersistStorageVtbl
PersistStorage_Vtbl
=
{
PersistStorage_QueryInterface
,
PersistStorage_AddRef
,
PersistStorage_Release
,
PersistStorage_GetClassID
,
PersistStorage_IsDirty
,
PersistStorage_InitNew
,
PersistStorage_Load
,
PersistStorage_Save
,
PersistStorage_SaveCompleted
,
PersistStorage_HandsOffStorage
};
static
HRESULT
WINAPI
PackageCF_QueryInterface
(
IClassFactory
*
iface
,
REFIID
riid
,
void
**
obj
)
{
TRACE
(
"(static)->(%s, %p)
\n
"
,
debugstr_guid
(
riid
),
obj
);
...
...
@@ -305,6 +579,7 @@ static HRESULT WINAPI PackageCF_CreateInstance(IClassFactory *iface, IUnknown *o
return
E_OUTOFMEMORY
;
package
->
IOleObject_iface
.
lpVtbl
=
&
OleObject_Vtbl
;
package
->
IPersistStorage_iface
.
lpVtbl
=
&
PersistStorage_Vtbl
;
return
IOleObject_QueryInterface
(
&
package
->
IOleObject_iface
,
iid
,
obj
);
}
...
...
dlls/packager/tests/Makefile.in
0 → 100644
View file @
d03c1cfc
TESTDLL
=
packager.dll
IMPORTS
=
ole32
C_SRCS
=
\
oleobj.c
dlls/packager/tests/oleobj.c
0 → 100644
View file @
d03c1cfc
/*
* Copyright 2014 Andrew Eikum for CodeWeavers
*
* 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 <initguid.h>
#include <olectl.h>
#include "wine/test.h"
DEFINE_GUID
(
CLSID_Package
,
0xf20da720
,
0xc02f
,
0x11ce
,
0x92
,
0x7b
,
0x08
,
0x00
,
0x09
,
0x5a
,
0xe3
,
0x40
);
DEFINE_GUID
(
CLSID_Package_Alt
,
0x0003000C
,
0x0000
,
0x0000
,
0xC0
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x00
,
0x46
);
static
HRESULT
WINAPI
ole10native_stream_QueryInterface
(
IStream
*
This
,
REFIID
riid
,
void
**
ppvObject
)
{
ok
(
0
,
"queryinterface
\n
"
);
return
E_NOTIMPL
;
}
static
ULONG
WINAPI
ole10native_stream_AddRef
(
IStream
*
This
)
{
return
2
;
}
static
ULONG
WINAPI
ole10native_stream_Release
(
IStream
*
This
)
{
return
1
;
}
static
UINT
offs
=
0
;
static
HRESULT
WINAPI
ole10native_stream_Read
(
IStream
*
This
,
void
*
pv
,
ULONG
cb
,
ULONG
*
pcbRead
)
{
ULONG
to_read
;
static
BYTE
data
[]
=
{
0xa5
,
0x00
,
0x00
,
0x00
,
/* total size */
0x02
,
0x00
,
/* ?? */
'l'
,
'a'
,
'b'
,
'e'
,
'l'
,
'.'
,
't'
,
'x'
,
't'
,
0
,
/* label? */
'f'
,
'i'
,
'l'
,
'e'
,
'n'
,
'a'
,
'm'
,
'e'
,
'.'
,
't'
,
'x'
,
't'
,
0
,
/* original filename? */
0x00
,
0x00
,
/* ?? */
0x03
,
0x00
,
/* ?? */
0x10
,
0x00
,
0x00
,
0x00
,
/* ASCIIZ filename size */
'C'
,
':'
,
'\\'
,
'f'
,
'i'
,
'l'
,
'e'
,
'n'
,
'a'
,
'm'
,
'e'
,
'2'
,
'.'
,
't'
,
'x'
,
't'
,
/* ASCIIZ filename */
0x0a
,
0x00
,
0x00
,
0x00
,
/* payload size */
's'
,
'o'
,
'm'
,
'e'
,
' '
,
't'
,
'e'
,
'x'
,
't'
,
'\n'
,
/* payload */
0x10
,
0x00
,
0x00
,
0x00
,
/* WCHAR filename size */
'C'
,
0
,
':'
,
0
,
'\\'
,
0
,
'f'
,
0
,
'i'
,
0
,
'l'
,
0
,
'e'
,
0
,
'n'
,
0
,
'a'
,
0
,
'm'
,
0
,
'e'
,
0
,
'3'
,
0
,
'.'
,
0
,
't'
,
0
,
'x'
,
0
,
't'
,
0
,
/* WCHAR filename */
0x0d
,
0x00
,
0x00
,
0x00
,
/* another filename size */
'f'
,
0
,
'i'
,
0
,
'l'
,
0
,
'e'
,
0
,
'n'
,
0
,
'a'
,
0
,
'm'
,
0
,
'e'
,
0
,
'4'
,
0
,
'.'
,
0
,
't'
,
0
,
'x'
,
0
,
't'
,
0
,
/* another filename */
0x10
,
0x00
,
0x00
,
0x00
,
/* another filename size */
'C'
,
0
,
':'
,
0
,
'\\'
,
0
,
'f'
,
0
,
'i'
,
0
,
'l'
,
0
,
'e'
,
0
,
'n'
,
0
,
'a'
,
0
,
'm'
,
0
,
'e'
,
0
,
'5'
,
0
,
'.'
,
0
,
't'
,
0
,
'x'
,
0
,
't'
,
0
,
/* another filename */
};
to_read
=
min
(
sizeof
(
data
)
-
offs
,
cb
);
memcpy
(
pv
,
data
+
offs
,
to_read
);
if
(
pcbRead
)
*
pcbRead
=
to_read
;
offs
+=
to_read
;
return
cb
==
to_read
?
S_OK
:
S_FALSE
;
}
static
HRESULT
WINAPI
ole10native_stream_Write
(
IStream
*
This
,
const
void
*
pv
,
ULONG
cb
,
ULONG
*
pcbWritten
)
{
ok
(
0
,
"write
\n
"
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
ole10native_stream_Seek
(
IStream
*
This
,
LARGE_INTEGER
dlibMove
,
DWORD
dwOrigin
,
ULARGE_INTEGER
*
plibNewPosition
)
{
if
(
dwOrigin
==
STREAM_SEEK_CUR
)
offs
+=
dlibMove
.
u
.
LowPart
;
else
if
(
dwOrigin
==
STREAM_SEEK_SET
)
offs
=
dlibMove
.
u
.
LowPart
;
if
(
plibNewPosition
){
plibNewPosition
->
u
.
HighPart
=
0
;
plibNewPosition
->
u
.
LowPart
=
offs
;
}
return
S_OK
;
}
static
HRESULT
WINAPI
ole10native_stream_SetSize
(
IStream
*
This
,
ULARGE_INTEGER
libNewSize
)
{
ok
(
0
,
"setsize
\n
"
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
ole10native_stream_CopyTo
(
IStream
*
This
,
IStream
*
pstm
,
ULARGE_INTEGER
cb
,
ULARGE_INTEGER
*
pcbRead
,
ULARGE_INTEGER
*
pcbWritten
)
{
ok
(
0
,
"copyto
\n
"
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
ole10native_stream_Commit
(
IStream
*
This
,
DWORD
grfCommitFlags
)
{
ok
(
0
,
"commit
\n
"
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
ole10native_stream_Revert
(
IStream
*
This
)
{
ok
(
0
,
"revert
\n
"
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
ole10native_stream_LockRegion
(
IStream
*
This
,
ULARGE_INTEGER
libOffset
,
ULARGE_INTEGER
cb
,
DWORD
dwLockType
)
{
ok
(
0
,
"lockregion
\n
"
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
ole10native_stream_UnlockRegion
(
IStream
*
This
,
ULARGE_INTEGER
libOffset
,
ULARGE_INTEGER
cb
,
DWORD
dwLockType
)
{
ok
(
0
,
"unlockregion
\n
"
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
ole10native_stream_Stat
(
IStream
*
This
,
STATSTG
*
pstatstg
,
DWORD
grfStatFlag
)
{
ok
(
0
,
"stat
\n
"
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
ole10native_stream_Clone
(
IStream
*
This
,
IStream
**
ppstm
)
{
ok
(
0
,
"clone
\n
"
);
return
E_NOTIMPL
;
}
static
IStreamVtbl
ole10native_stream_vtbl
=
{
ole10native_stream_QueryInterface
,
ole10native_stream_AddRef
,
ole10native_stream_Release
,
ole10native_stream_Read
,
ole10native_stream_Write
,
ole10native_stream_Seek
,
ole10native_stream_SetSize
,
ole10native_stream_CopyTo
,
ole10native_stream_Commit
,
ole10native_stream_Revert
,
ole10native_stream_LockRegion
,
ole10native_stream_UnlockRegion
,
ole10native_stream_Stat
,
ole10native_stream_Clone
};
static
IStream
ole10native_stream
=
{
&
ole10native_stream_vtbl
};
HRESULT
WINAPI
stg_QueryInterface
(
IStorage
*
This
,
REFIID
riid
,
void
**
ppvObject
)
{
ok
(
0
,
"queryinterface: %s
\n
"
,
wine_dbgstr_guid
(
riid
));
if
(
IsEqualIID
(
riid
,
&
IID_IUnknown
)
||
IsEqualIID
(
riid
,
&
IID_IStorage
)){
*
ppvObject
=
This
;
return
S_OK
;
}
return
E_NOINTERFACE
;
}
static
ULONG
WINAPI
stg_AddRef
(
IStorage
*
This
)
{
return
2
;
}
static
ULONG
WINAPI
stg_Release
(
IStorage
*
This
)
{
return
1
;
}
static
HRESULT
WINAPI
stg_CreateStream
(
IStorage
*
This
,
LPCOLESTR
pwcsName
,
DWORD
grfMode
,
DWORD
reserved1
,
DWORD
reserved2
,
IStream
**
ppstm
)
{
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_OpenStream
(
IStorage
*
This
,
LPCOLESTR
pwcsName
,
void
*
reserved1
,
DWORD
grfMode
,
DWORD
reserved2
,
IStream
**
ppstm
)
{
static
const
WCHAR
ole10NativeW
[]
=
{
1
,
'O'
,
'l'
,
'e'
,
'1'
,
'0'
,
'N'
,
'a'
,
't'
,
'i'
,
'v'
,
'e'
,
0
};
if
(
lstrcmpW
(
pwcsName
,
ole10NativeW
))
return
STG_E_FILENOTFOUND
;
*
ppstm
=
&
ole10native_stream
;
return
S_OK
;
}
HRESULT
WINAPI
stg_CreateStorage
(
IStorage
*
This
,
LPCOLESTR
pwcsName
,
DWORD
grfMode
,
DWORD
dwStgFmt
,
DWORD
reserved2
,
IStorage
**
ppstg
)
{
ok
(
0
,
"createstorage
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_OpenStorage
(
IStorage
*
This
,
LPCOLESTR
pwcsName
,
IStorage
*
pstgPriority
,
DWORD
grfMode
,
SNB
snbExclude
,
DWORD
reserved
,
IStorage
**
ppstg
)
{
ok
(
0
,
"openstorage: %s
\n
"
,
wine_dbgstr_w
(
pwcsName
));
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_CopyTo
(
IStorage
*
This
,
DWORD
ciidExclude
,
const
IID
*
rgiidExclude
,
SNB
snbExclude
,
IStorage
*
pstgDest
)
{
ok
(
0
,
"copyto
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_MoveElementTo
(
IStorage
*
This
,
LPCOLESTR
pwcsName
,
IStorage
*
pstgDest
,
LPCOLESTR
pwcsNewName
,
DWORD
grfFlags
)
{
ok
(
0
,
"moveelem
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_Commit
(
IStorage
*
This
,
DWORD
grfCommitFlags
)
{
ok
(
0
,
"commit
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_Revert
(
IStorage
*
This
)
{
ok
(
0
,
"revert
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_EnumElements
(
IStorage
*
This
,
DWORD
reserved1
,
void
*
reserved2
,
DWORD
reserved3
,
IEnumSTATSTG
**
ppenum
)
{
ok
(
0
,
"enumelem
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_DestroyElement
(
IStorage
*
This
,
LPCOLESTR
pwcsName
)
{
ok
(
0
,
"destroyelem
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_RenameElement
(
IStorage
*
This
,
LPCOLESTR
pwcsOldName
,
LPCOLESTR
pwcsNewName
)
{
ok
(
0
,
"renamelem
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_SetElementTimes
(
IStorage
*
This
,
LPCOLESTR
pwcsName
,
const
FILETIME
*
pctime
,
const
FILETIME
*
patime
,
const
FILETIME
*
pmtime
)
{
ok
(
0
,
"setelem
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_SetClass
(
IStorage
*
This
,
REFCLSID
clsid
)
{
ok
(
0
,
"setclass
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_SetStateBits
(
IStorage
*
This
,
DWORD
grfStateBits
,
DWORD
grfMask
)
{
ok
(
0
,
"setstate
\n
"
);
return
E_NOTIMPL
;
}
HRESULT
WINAPI
stg_Stat
(
IStorage
*
This
,
STATSTG
*
pstatstg
,
DWORD
grfStatFlag
)
{
memset
(
pstatstg
,
0
,
sizeof
(
*
pstatstg
));
pstatstg
->
clsid
=
CLSID_Package
;
pstatstg
->
type
=
STGTY_STORAGE
;
return
S_OK
;
}
static
IStorageVtbl
stg_vtbl
=
{
stg_QueryInterface
,
stg_AddRef
,
stg_Release
,
stg_CreateStream
,
stg_OpenStream
,
stg_CreateStorage
,
stg_OpenStorage
,
stg_CopyTo
,
stg_MoveElementTo
,
stg_Commit
,
stg_Revert
,
stg_EnumElements
,
stg_DestroyElement
,
stg_RenameElement
,
stg_SetElementTimes
,
stg_SetClass
,
stg_SetStateBits
,
stg_Stat
,
};
static
IStorage
stg
=
{
&
stg_vtbl
};
static
void
test_packager
(
void
)
{
IOleObject
*
oleobj
;
IPersistStorage
*
persist
;
DWORD
len
,
bytes_read
;
HRESULT
hr
;
HANDLE
file
;
WCHAR
filename
[
MAX_PATH
];
char
contents
[
11
];
BOOL
br
,
extended
=
FALSE
;
static
const
WCHAR
filename3W
[]
=
{
'f'
,
'i'
,
'l'
,
'e'
,
'n'
,
'a'
,
'm'
,
'e'
,
'3'
,
'.'
,
't'
,
'x'
,
't'
,
0
};
hr
=
CoCreateInstance
(
&
CLSID_Package
,
NULL
,
CLSCTX_INPROC_SERVER
|
CLSCTX_INPROC_HANDLER
,
&
IID_IOleObject
,
(
void
**
)
&
oleobj
);
ok
(
hr
==
S_OK
||
hr
==
REGDB_E_CLASSNOTREG
,
"CoCreateInstance(CLSID_Package) failed: %08x
\n
"
,
hr
);
if
(
hr
==
S_OK
){
IOleObject_Release
(
oleobj
);
/* older OSes store temporary files in obscure locations, so don't run
* the full tests on them */
extended
=
TRUE
;
}
else
win_skip
(
"Older OS doesn't support modern packager
\n
"
);
hr
=
CoCreateInstance
(
&
CLSID_Package_Alt
,
NULL
,
CLSCTX_INPROC_SERVER
|
CLSCTX_INPROC_HANDLER
,
&
IID_IOleObject
,
(
void
**
)
&
oleobj
);
ok
(
hr
==
S_OK
,
"CoCreateInstance(CLSID_Package_Alt) failed: %08x
\n
"
,
hr
);
hr
=
IOleObject_QueryInterface
(
oleobj
,
&
IID_IPersistStorage
,
(
void
**
)
&
persist
);
ok
(
hr
==
S_OK
,
"QueryInterface(IPersistStorage) failed: %08x
\n
"
,
hr
);
hr
=
IPersistStorage_Load
(
persist
,
&
stg
);
ok
(
hr
==
S_OK
,
"Load failed: %08x
\n
"
,
hr
);
if
(
extended
){
len
=
GetTempPathW
(
sizeof
(
filename
)
/
sizeof
(
*
filename
),
filename
);
lstrcpyW
(
filename
+
len
,
filename3W
);
file
=
CreateFileW
(
filename
,
GENERIC_READ
,
0
,
NULL
,
OPEN_EXISTING
,
FILE_ATTRIBUTE_NORMAL
,
NULL
);
ok
(
file
!=
INVALID_HANDLE_VALUE
,
"Couldn't find temporary file %s: %u
\n
"
,
wine_dbgstr_w
(
filename
),
GetLastError
());
br
=
ReadFile
(
file
,
contents
,
sizeof
(
contents
),
&
bytes_read
,
NULL
);
ok
(
br
==
TRUE
,
"ReadFile failed
\n
"
);
ok
(
bytes_read
==
10
,
"Got wrong file size: %u
\n
"
,
bytes_read
);
ok
(
!
memcmp
(
contents
,
"some text
\n
"
,
10
),
"Got wrong file contents
\n
"
);
CloseHandle
(
file
);
}
IPersistStorage_Release
(
persist
);
IOleObject_Release
(
oleobj
);
}
START_TEST
(
oleobj
)
{
CoInitialize
(
NULL
);
test_packager
();
CoUninitialize
();
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment