404 lines
13 KiB
C++
404 lines
13 KiB
C++
#include "priv.h"
|
|
#include "privcpp.h"
|
|
|
|
|
|
// Constructor
|
|
CPackage_IPersistStorage::CPackage_IPersistStorage(CPackage *pPackage) : _pPackage(pPackage)
|
|
{
|
|
ASSERT(_cRef == 0);
|
|
ASSERT(_psState == PSSTATE_UNINIT);
|
|
}
|
|
|
|
|
|
CPackage_IPersistStorage::~CPackage_IPersistStorage()
|
|
{
|
|
DebugMsg(DM_TRACE, "CPackage_IPersistStorage destroyed with ref count %d", _cRef);
|
|
}
|
|
|
|
|
|
// IUnknown Methods...
|
|
HRESULT CPackage_IPersistStorage::QueryInterface(REFIID iid, void ** ppv)
|
|
{
|
|
return _pPackage->QueryInterface(iid, ppv); // delegate to CPackage
|
|
}
|
|
|
|
|
|
ULONG CPackage_IPersistStorage::AddRef(void)
|
|
{
|
|
_cRef++; // interface ref count for debug
|
|
return _pPackage->AddRef(); // delegate to CPackage
|
|
}
|
|
|
|
|
|
ULONG CPackage_IPersistStorage::Release(void)
|
|
{
|
|
_cRef--; // interface ref count for debug
|
|
return _pPackage->Release(); // delegate to CPackage
|
|
}
|
|
|
|
|
|
// IPersistStorage Methods...
|
|
HRESULT CPackage_IPersistStorage::GetClassID(LPCLSID pClassID)
|
|
{
|
|
DebugMsg(DM_TRACE, "pack ps - GetClassID() called.");
|
|
|
|
if (_psState == PSSTATE_UNINIT)
|
|
return E_UNEXPECTED;
|
|
|
|
if (pClassID == NULL)
|
|
return E_INVALIDARG;
|
|
|
|
*pClassID = CLSID_CPackage;
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
HRESULT CPackage_IPersistStorage::IsDirty(void)
|
|
{
|
|
DebugMsg(DM_TRACE, "pack ps - IsDirty() called. _fIsDirty==%d", (INT) _pPackage->_fIsDirty);
|
|
if (_psState == PSSTATE_UNINIT)
|
|
return E_UNEXPECTED;
|
|
|
|
return (_pPackage->_fIsDirty ? S_OK : S_FALSE);
|
|
}
|
|
|
|
|
|
HRESULT CPackage_IPersistStorage::InitNew(IStorage *pstg)
|
|
{
|
|
HRESULT hr;
|
|
|
|
DebugMsg(DM_TRACE, "pack ps - InitNew() called.");
|
|
|
|
if (_psState != PSSTATE_UNINIT)
|
|
return E_UNEXPECTED;
|
|
|
|
if (!pstg)
|
|
return E_POINTER;
|
|
|
|
// Create a stream to save the package and cache the pointer. By doing
|
|
// this now we ensure being able to save in low memory conditions.
|
|
hr = pstg->CreateStream(SZCONTENTS, STGM_DIRECT | STGM_CREATE | STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &_pPackage->_pstm);
|
|
if (FAILED(hr))
|
|
goto ErrRet;
|
|
|
|
hr = WriteFmtUserTypeStg(pstg, _pPackage->_cf, SZUSERTYPE);
|
|
if (FAILED(hr)) {
|
|
goto ErrRet;
|
|
}
|
|
|
|
_pPackage->_fIsDirty = TRUE;
|
|
_psState = PSSTATE_SCRIBBLE;
|
|
|
|
_pPackage->_pIStorage = pstg; // cache the IStorage pointer
|
|
_pPackage->_pIStorage->AddRef(); // but don't forget to addref it!
|
|
|
|
DebugMsg(DM_TRACE, " leaving InitNew()");
|
|
return S_OK;
|
|
|
|
ErrRet:
|
|
if (_pPackage->_pstm) {
|
|
_pPackage->_pstm->Release();
|
|
_pPackage->_pstm = NULL;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT CPackage_IPersistStorage::Load(IStorage *pstg)
|
|
{
|
|
HRESULT hr;
|
|
LPSTREAM pstm = NULL; // package contents
|
|
CLSID clsid;
|
|
|
|
DebugMsg(DM_TRACE, "pack ps - Load() called.");
|
|
|
|
if (_psState != PSSTATE_UNINIT) {
|
|
DebugMsg(DM_TRACE, " wrong state!!");
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
if (!pstg) {
|
|
DebugMsg(DM_TRACE, " bad pointer!!");
|
|
return E_POINTER;
|
|
}
|
|
|
|
// check to make sure this is one of our storages
|
|
hr = ReadClassStg(pstg, &clsid);
|
|
if (SUCCEEDED(hr) && (clsid != CLSID_CPackage && clsid != CLSID_OldPackage) || FAILED(hr))
|
|
{
|
|
DebugMsg(DM_TRACE, " bad storage type!!");
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
hr = pstg->OpenStream(SZCONTENTS, 0, STGM_DIRECT | STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &pstm);
|
|
if (FAILED(hr)) {
|
|
DebugMsg(DM_TRACE, " couldn't open contents stream!!");
|
|
DebugMsg(DM_TRACE, " hr==%Xh", hr);
|
|
goto ErrRet;
|
|
}
|
|
|
|
hr = _pPackage->PackageReadFromStream(pstm);
|
|
if (FAILED(hr))
|
|
goto ErrRet;
|
|
|
|
_pPackage->_pstm = pstm;
|
|
_pPackage->_pIStorage = pstg;
|
|
pstg->AddRef();
|
|
|
|
_psState = PSSTATE_SCRIBBLE;
|
|
_pPackage->_fIsDirty = FALSE;
|
|
_pPackage->_fLoaded = TRUE;
|
|
|
|
DebugMsg(DM_TRACE, " leaving Load()");
|
|
return S_OK;
|
|
|
|
ErrRet:
|
|
if (pstm)
|
|
pstm->Release();
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT CPackage_IPersistStorage::Save(IStorage *pstg, BOOL fSameAsLoad)
|
|
{
|
|
HRESULT hr;
|
|
LPSTREAM pstm = NULL;
|
|
|
|
DebugMsg(DM_TRACE, "pack ps - Save() called.");
|
|
|
|
// must come here from scribble state
|
|
if ((_psState != PSSTATE_SCRIBBLE) && fSameAsLoad) {
|
|
DebugMsg(DM_TRACE, " bad state!!");
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// must have an IStorage if not SameAsLoad
|
|
if (!pstg && !fSameAsLoad) {
|
|
DebugMsg(DM_TRACE, " bad pointer!!");
|
|
return E_POINTER;
|
|
}
|
|
|
|
// hopefully, the container calls WriteClassStg with our CLSID before
|
|
// we get here, that way we can overwrite that and write out the old
|
|
// packager's CLSID so that the old packager can read new packages.
|
|
if (FAILED(WriteClassStg(pstg, CLSID_OldPackage))) {
|
|
DebugMsg(DM_TRACE, " couldn't write CLSID to storage!!");
|
|
return E_FAIL;
|
|
}
|
|
|
|
// ok, we have four possible ways we could be calling Save:
|
|
// 1. we're creating a new package and saving to the same storage we received in InitNew
|
|
// 2. We're creating a new package and saving to a different storage than we received in InitNew
|
|
// 3. We were loaded by a container and we're saving to the same stream we received in Load
|
|
// 4. We were loaded by a container and we're saving to a different stream than we received in Load
|
|
|
|
// Same Storage as Load
|
|
if (fSameAsLoad) {
|
|
DebugMsg(DM_TRACE, " Same as load.");
|
|
|
|
LARGE_INTEGER li = { 0, 0 };
|
|
pstm = _pPackage->_pstm;
|
|
|
|
// If we're not dirty, so there's nothing new to save.
|
|
if (FALSE == _pPackage->_fIsDirty) {
|
|
DebugMsg(DM_TRACE, " not saving cause we're not dirty!!");
|
|
return S_OK;
|
|
}
|
|
|
|
// if we are dirty, set the seek pointer to the beginning and write
|
|
// the package information to the stream
|
|
hr = pstm->Seek(li, STREAM_SEEK_SET, NULL);
|
|
if (FAILED(hr)) {
|
|
DebugMsg(DM_TRACE, " couldn't set contents pointer!!");
|
|
return hr;
|
|
}
|
|
|
|
// case 1: new package
|
|
if (!_pPackage->_fLoaded) {
|
|
switch (_pPackage->_panetype) {
|
|
LPTSTR temp;
|
|
case PEMBED:
|
|
// if haven't created a temp file yet, then use the the
|
|
// file to be packaged to get our file contents from,
|
|
// otherwise we just use the temp file, because if we
|
|
// have a temp file, it contains the most recent info.
|
|
temp = _pPackage->_pEmbed->pszTempName;
|
|
|
|
if (!_pPackage->_pEmbed->pszTempName) {
|
|
DebugMsg(DM_TRACE, " case 1a:not loaded, using initFile.");
|
|
_pPackage->_pEmbed->pszTempName = _pPackage->_pEmbed->fd.cFileName;
|
|
}
|
|
else {
|
|
DebugMsg(DM_TRACE, " case 1b:not loaded, using tempfile.");
|
|
}
|
|
|
|
hr = _pPackage->PackageWriteToStream(pstm);
|
|
|
|
// reset our temp name back, since we might have changed it
|
|
// basically, this just sets it to NULL if it was before
|
|
_pPackage->_pEmbed->pszTempName = temp;
|
|
break;
|
|
case CMDLINK:
|
|
// nothing screwy to do here...just write out the info which we already have in memory.
|
|
hr = _pPackage->PackageWriteToStream(pstm);
|
|
break;
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
return hr;
|
|
}
|
|
else {// case 3: loaded package
|
|
hr = _pPackage->PackageWriteToStream(pstm);
|
|
}
|
|
}
|
|
else {// NEW Storage
|
|
DebugMsg(DM_TRACE, " NOT same as load.");
|
|
hr = pstg->CreateStream(SZCONTENTS, STGM_DIRECT | STGM_CREATE | STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, 0, &pstm);
|
|
if (FAILED(hr)) {
|
|
DebugMsg(DM_TRACE, " couldn't create contents stream!!");
|
|
return hr;
|
|
}
|
|
WriteFmtUserTypeStg(pstg, _pPackage->_cf, SZUSERTYPE);
|
|
|
|
// case 2:
|
|
if (!_pPackage->_fLoaded) {
|
|
switch (_pPackage->_panetype) {
|
|
LPTSTR temp;
|
|
case PEMBED:
|
|
temp = _pPackage->_pEmbed->pszTempName;
|
|
|
|
if (!_pPackage->_pEmbed->pszTempName) {
|
|
DebugMsg(DM_TRACE, " case 2a:not loaded, using initFile.");
|
|
_pPackage->_pEmbed->pszTempName = _pPackage->_pEmbed->fd.cFileName;
|
|
}
|
|
else {
|
|
DebugMsg(DM_TRACE, " case 2b:not loaded, using tempfile.");
|
|
}
|
|
|
|
hr = _pPackage->PackageWriteToStream(pstm);
|
|
|
|
// reset our temp name back, since we might have changed it
|
|
// basically, this just sets it to NULL if it was before
|
|
_pPackage->_pEmbed->pszTempName = temp;
|
|
break;
|
|
case CMDLINK:
|
|
// nothing interesting to do here, other than write out the package.
|
|
hr = _pPackage->PackageWriteToStream(pstm);
|
|
break;
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
goto ErrRet;
|
|
}
|
|
|
|
// case 4:
|
|
else {
|
|
if (_pPackage->_panetype == PEMBED && _pPackage->_pEmbed->pszTempName) {
|
|
DebugMsg(DM_TRACE, " case 4a:loaded, using tempfile.");
|
|
hr = _pPackage->PackageWriteToStream(pstm);
|
|
if (FAILED(hr))
|
|
goto ErrRet;
|
|
}
|
|
else {
|
|
DebugMsg(DM_TRACE, " case 4b:loaded, using stream.");
|
|
ULARGE_INTEGER uli = { 0xFFFFFFFF, 0xFFFFFFFF };
|
|
LARGE_INTEGER li = { 0, 0 };
|
|
|
|
hr = _pPackage->_pstm->Seek(li, STREAM_SEEK_SET, NULL);
|
|
if (FAILED(hr))
|
|
goto ErrRet;
|
|
|
|
hr = _pPackage->_pstm->CopyTo(pstm, uli, NULL, NULL);
|
|
if (FAILED(hr))
|
|
goto ErrRet;
|
|
}
|
|
}
|
|
ErrRet:
|
|
UINT u = pstm->Release();
|
|
DebugMsg(DM_TRACE, " new stream released. cRef==%d.", u);
|
|
if (FAILED(hr))
|
|
return hr;
|
|
}
|
|
|
|
_psState = PSSTATE_ZOMBIE;
|
|
|
|
DebugMsg(DM_TRACE, " leaving Save()");
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
HRESULT CPackage_IPersistStorage::SaveCompleted(IStorage *pstg)
|
|
{
|
|
HRESULT hr;
|
|
LPSTREAM pstm;
|
|
|
|
DebugMsg(DM_TRACE, "pack ps - SaveCompleted() called.");
|
|
|
|
// must be called from no-scribble or hands-off state
|
|
if (!(_psState == PSSTATE_ZOMBIE || _psState == PSSTATE_HANDSOFF))
|
|
return E_UNEXPECTED;
|
|
|
|
// if we're hands-off, we'd better get a storage to re-init from
|
|
if (!pstg && _psState == PSSTATE_HANDSOFF)
|
|
return E_UNEXPECTED;
|
|
|
|
// if pstg is NULL then we have everything we need, otherwise we must release our held pointers and reinitialize.
|
|
if (pstg != NULL) {
|
|
DebugMsg(DM_TRACE, " getting new storage.");
|
|
hr = pstg->OpenStream(SZCONTENTS, 0, STGM_DIRECT | STGM_READWRITE | STGM_SHARE_EXCLUSIVE, 0, &pstm);
|
|
if (FAILED(hr))
|
|
return hr;
|
|
|
|
if (_pPackage->_pstm != NULL)
|
|
_pPackage->_pstm->Release();
|
|
|
|
// this will be reinitialized when it's needed
|
|
if (_pPackage->_pstmFileContents != NULL) {
|
|
_pPackage->_pstmFileContents->Release();
|
|
_pPackage->_pstmFileContents = NULL;
|
|
}
|
|
|
|
if (_pPackage->_pIStorage != NULL)
|
|
_pPackage->_pIStorage->Release();
|
|
|
|
_pPackage->_pstm = pstm;
|
|
_pPackage->_pIStorage = pstg;
|
|
_pPackage->_pIStorage->AddRef();
|
|
}
|
|
|
|
_psState = PSSTATE_SCRIBBLE;
|
|
_pPackage->_fIsDirty = FALSE;
|
|
DebugMsg(DM_TRACE, " leaving SaveCompleted()");
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
HRESULT CPackage_IPersistStorage::HandsOffStorage(void)
|
|
{
|
|
DebugMsg(DM_TRACE, "pack ps - HandsOffStorage() called.");
|
|
|
|
// must come from scribble or no-scribble. a repeated call to
|
|
// HandsOffStorage is an unexpected error (bug in client).
|
|
if (_psState == PSSTATE_UNINIT || _psState == PSSTATE_HANDSOFF)
|
|
return E_UNEXPECTED;
|
|
|
|
// release our held pointers
|
|
if (_pPackage->_pstmFileContents != NULL) {
|
|
_pPackage->_pstmFileContents->Release();
|
|
_pPackage->_pstmFileContents = NULL;
|
|
}
|
|
|
|
if (_pPackage->_pstm != NULL) {
|
|
_pPackage->_pstm->Release();
|
|
_pPackage->_pstm = NULL;
|
|
}
|
|
|
|
if (_pPackage->_pIStorage != NULL) {
|
|
_pPackage->_pIStorage->Release();
|
|
_pPackage->_pIStorage = NULL;
|
|
}
|
|
|
|
_psState = PSSTATE_HANDSOFF;
|
|
DebugMsg(DM_TRACE, " leaving HandsOffStorage()");
|
|
return S_OK;
|
|
} |