blob: 1bca1d1c928ae73b5d9db0c2b32d5145ac36b67f [file] [log] [blame]
/**************************************************************
*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_tools.hxx"
#include <tools/debug.hxx>
#include <tools/pstm.hxx>
#define STOR_NO_OPTIMIZE
/***********************************************************************/
/************************************************************************
|* SvClassManager::Register()
*************************************************************************/
void SvClassManager::Register( sal_uInt16 nClassId, SvCreateInstancePersist pFunc )
{
#ifdef DBG_UTIL
SvCreateInstancePersist p;
p = Get( nClassId );
DBG_ASSERT( !p || p == pFunc, "register class with same id" );
#endif
aAssocTable.insert(Map::value_type(nClassId, pFunc));
}
/************************************************************************
|* SvClassManager::Get()
*************************************************************************/
SvCreateInstancePersist SvClassManager::Get( sal_uInt16 nClassId )
{
Map::const_iterator i(aAssocTable.find(nClassId));
return i == aAssocTable.end() ? 0 : i->second;
}
/****************** SvRttiBase *******************************************/
TYPEINIT0( SvRttiBase );
/****************** SvPersistBaseMemberList ******************************/
SvPersistBaseMemberList::SvPersistBaseMemberList(){}
SvPersistBaseMemberList::SvPersistBaseMemberList(
sal_uInt16 nInitSz, sal_uInt16 nResize )
: SuperSvPersistBaseMemberList( nInitSz, nResize ){}
#define PERSIST_LIST_VER (sal_uInt8)0
#define PERSIST_LIST_DBGUTIL (sal_uInt8)0x80
/************************************************************************
|* SvPersistBaseMemberList::WriteOnlyStreamedObjects()
*************************************************************************/
void SvPersistBaseMemberList::WriteObjects( SvPersistStream & rStm,
sal_Bool bOnlyStreamed ) const
{
#ifdef STOR_NO_OPTIMIZE
rStm << (sal_uInt8)(PERSIST_LIST_VER | PERSIST_LIST_DBGUTIL);
sal_uInt32 nObjPos = rStm.WriteDummyLen();
#else
sal_uInt8 bTmp = PERSIST_LIST_VER;
rStm << bTmp;
#endif
sal_uInt32 nCountMember = Count();
sal_uIntPtr nCountPos = rStm.Tell();
sal_uInt32 nWriteCount = 0;
rStm << nCountMember;
//bloss die Liste nicht veraendern,
//wegen Seiteneffekten beim Save
for( sal_uIntPtr n = 0; n < nCountMember; n++ )
{
SvPersistBase * pObj = GetObject( n );
if( !bOnlyStreamed || rStm.IsStreamed( pObj ) )
{ // Objekt soll geschrieben werden
rStm << GetObject( n );
nWriteCount++;
}
}
if( nWriteCount != nCountMember )
{
// nicht alle Objekte geschrieben, Count anpassen
sal_uIntPtr nPos = rStm.Tell();
rStm.Seek( nCountPos );
rStm << nWriteCount;
rStm.Seek( nPos );
}
#ifdef STOR_NO_OPTIMIZE
rStm.WriteLen( nObjPos );
#endif
}
/************************************************************************
|* operator << ()
*************************************************************************/
SvPersistStream& operator << ( SvPersistStream & rStm,
const SvPersistBaseMemberList & rLst )
{
rLst.WriteObjects( rStm );
return rStm;
}
/************************************************************************
|* operator >> ()
*************************************************************************/
SvPersistStream& operator >> ( SvPersistStream & rStm,
SvPersistBaseMemberList & rLst )
{
sal_uInt8 nVer;
rStm >> nVer;
if( (nVer & ~PERSIST_LIST_DBGUTIL) != PERSIST_LIST_VER )
{
rStm.SetError( SVSTREAM_GENERALERROR );
DBG_ERROR( "persist list, false version" );
}
sal_uInt32 nObjLen(0), nObjPos(0);
if( nVer & PERSIST_LIST_DBGUTIL )
nObjLen = rStm.ReadLen( &nObjPos );
sal_uInt32 nCount;
rStm >> nCount;
for( sal_uIntPtr n = 0; n < nCount && rStm.GetError() == SVSTREAM_OK; n++ )
{
SvPersistBase * pObj;
rStm >> pObj;
if( pObj )
rLst.Append( pObj );
}
#ifdef DBG_UTIL
if( nObjLen + nObjPos != rStm.Tell() )
{
ByteString aStr( "false list len: read = " );
aStr += ByteString::CreateFromInt32( (long)(rStm.Tell() - nObjPos) );
aStr += ", should = ";
aStr += ByteString::CreateFromInt64(nObjLen);
DBG_ERROR( aStr.GetBuffer() );
}
#endif
return rStm;
}
//=========================================================================
SvPersistStream::SvPersistStream
(
SvClassManager & rMgr, /* Alle Factorys, deren Objekt geladen und
gespeichert werdn k"onnen */
SvStream * pStream, /* Dieser Stream wird als Medium genommen, auf
dem der PersistStream arbeitet */
sal_uInt32 nStartIdxP /* Ab diesem Index werden die Id's f"ur
die Objekte vergeben, er muss gr"osser
als Null sein. */
)
: rClassMgr( rMgr )
, pStm( pStream )
, aPUIdx( nStartIdxP )
, nStartIdx( nStartIdxP )
, pRefStm( NULL )
, nFlags( 0 )
/* [Beschreibung]
Der Konstruktor der Klasse SvPersistStream. Die Objekte rMgr und
pStream d"urfen nicht ver"andert werden, solange sie in einem
SvPersistStream eingesetzt sind. Eine Aussnahme gibt es f"ur
pStream (siehe <SvPersistStream::SetStream>).
*/
{
DBG_ASSERT( nStartIdx != 0, "zero index not allowed" );
bIsWritable = sal_True;
if( pStm )
{
SetVersion( pStm->GetVersion() );
SetError( pStm->GetError() );
SyncSvStream( pStm->Tell() );
}
}
//=========================================================================
SvPersistStream::SvPersistStream
(
SvClassManager & rMgr, /* Alle Factorys, deren Objekt geladen und
gespeichert werdn k"onnen */
SvStream * pStream, /* Dieser Stream wird als Medium genommen, auf
dem der PersistStream arbeitet */
const SvPersistStream & rPersStm
/* Wenn PersistStream's verschachtelt werden,
dann ist dies der Parent-Stream. */
)
: rClassMgr( rMgr )
, pStm( pStream )
// Bereiche nicht ueberschneiden, deshalb nur groessere Indexe
, aPUIdx( rPersStm.GetCurMaxIndex() +1 )
, nStartIdx( rPersStm.GetCurMaxIndex() +1 )
, pRefStm( &rPersStm )
, nFlags( 0 )
/* [Beschreibung]
Der Konstruktor der Klasse SvPersistStream. Die Objekte rMgr und
pStream d"urfen nicht ver"andert werden, solange sie in einem
SvPersistStream eingesetzt sind. Eine Aussnahme gibt es f"ur
pStream (siehe <SvPersistStream::SetStream>).
Durch diesen Konstruktor wird eine Hierarchiebildung unterst"utzt.
Alle Objekte aus einer Hierarchie m"ussen erst geladen werden,
wenn das erste aus dieser Hierarchie benutzt werden soll.
*/
{
bIsWritable = sal_True;
if( pStm )
{
SetVersion( pStm->GetVersion() );
SetError( pStm->GetError() );
SyncSvStream( pStm->Tell() );
}
}
//=========================================================================
SvPersistStream::~SvPersistStream()
/* [Beschreibung]
Der Detruktor ruft die Methode <SvPersistStream::SetStream>
mit NULL.
*/
{
SetStream( NULL );
}
//=========================================================================
void SvPersistStream::SetStream
(
SvStream * pStream /* auf diesem Stream arbeitet der PersistStream */
)
/* [Beschreibung]
Es wird ein Medium (pStream) eingesetzt, auf dem PersistStream arbeitet.
Dieses darf nicht von aussen modifiziert werden, solange es
eingesetzt ist. Es sei denn, w"ahrend auf dem Medium gearbeitet
wird, wird keine Methode von SvPersistStream gerufen, bevor
nicht <SvPersistStream::SetStream> mit demselben Medium gerufen
wurde.
*/
{
if( pStm != pStream )
{
if( pStm )
{
SyncSysStream();
pStm->SetError( GetError() );
}
pStm = pStream;
}
if( pStm )
{
SetVersion( pStm->GetVersion() );
SetError( pStm->GetError() );
SyncSvStream( pStm->Tell() );
}
}
//=========================================================================
sal_uInt16 SvPersistStream::IsA() const
/* [Beschreibung]
Gibt den Identifier dieses Streamklasse zur"uck.
[R"uckgabewert]
sal_uInt16 ID_PERSISTSTREAM wird zur"uckgegeben.
[Querverweise]
<SvStream::IsA>
*/
{
return ID_PERSISTSTREAM;
}
/*************************************************************************
|* SvPersistStream::ResetError()
*************************************************************************/
void SvPersistStream::ResetError()
{
SvStream::ResetError();
DBG_ASSERT( pStm, "stream not set" );
pStm->ResetError();
}
/*************************************************************************
|* SvPersistStream::GetData()
*************************************************************************/
sal_uIntPtr SvPersistStream::GetData( void* pData, sal_uIntPtr nSize )
{
DBG_ASSERT( pStm, "stream not set" );
sal_uIntPtr nRet = pStm->Read( pData, nSize );
SetError( pStm->GetError() );
return nRet;
}
/*************************************************************************
|* SvPersistStream::PutData()
*************************************************************************/
sal_uIntPtr SvPersistStream::PutData( const void* pData, sal_uIntPtr nSize )
{
DBG_ASSERT( pStm, "stream not set" );
sal_uIntPtr nRet = pStm->Write( pData, nSize );
SetError( pStm->GetError() );
return nRet;
}
/*************************************************************************
|* SvPersistStream::Seek()
*************************************************************************/
sal_uIntPtr SvPersistStream::SeekPos( sal_uIntPtr nPos )
{
DBG_ASSERT( pStm, "stream not set" );
sal_uIntPtr nRet = pStm->Seek( nPos );
SetError( pStm->GetError() );
return nRet;
}
/*************************************************************************
|* SvPersistStream::FlushData()
*************************************************************************/
void SvPersistStream::FlushData()
{
}
/*************************************************************************
|* SvPersistStream::GetCurMaxIndex()
*************************************************************************/
sal_uIntPtr SvPersistStream::GetCurMaxIndex( const SvPersistUIdx & rIdx ) const
{
// const bekomme ich nicht den hoechsten Index
SvPersistUIdx * p = (SvPersistUIdx *)&rIdx;
// alten merken
sal_uIntPtr nCurIdx = p->GetCurIndex();
p->Last();
// Bereiche nicht ueberschneiden, deshalb nur groessere Indexe
sal_uIntPtr nMaxIdx = p->GetCurIndex();
// wieder herstellen
p->Seek( nCurIdx );
return nMaxIdx;
}
/*************************************************************************
|* SvPersistStream::GetIndex()
*************************************************************************/
sal_uIntPtr SvPersistStream::GetIndex( SvPersistBase * pObj ) const
{
sal_uIntPtr nId = (sal_uIntPtr)aPTable.Get( (sal_uIntPtr)pObj );
if( !nId && pRefStm )
return pRefStm->GetIndex( pObj );
return nId;
}
/*************************************************************************
|* SvPersistStream::GetObject)
*************************************************************************/
SvPersistBase * SvPersistStream::GetObject( sal_uIntPtr nIdx ) const
{
if( nIdx >= nStartIdx )
return aPUIdx.Get( nIdx );
else if( pRefStm )
return pRefStm->GetObject( nIdx );
return NULL;
}
//=========================================================================
#define LEN_1 0x80
#define LEN_2 0x40
#define LEN_4 0x20
#define LEN_5 0x10
sal_uInt32 SvPersistStream::ReadCompressed
(
SvStream & rStm /* Aus diesem Stream werden die komprimierten Daten
gelesen */
)
/* [Beschreibung]
Ein im Stream komprimiert abgelegtes Wort wird gelesen. In welchem
Format komprimiert wird, siehe <SvPersistStream::WriteCompressed>.
[R"uckgabewert]
sal_uInt32 Das nicht komprimierte Wort wird zur"uckgegeben.
[Querverweise]
*/
{
sal_uInt32 nRet(0);
sal_uInt8 nMask;
rStm >> nMask;
if( nMask & LEN_1 )
nRet = ~LEN_1 & nMask;
else if( nMask & LEN_2 )
{
nRet = ~LEN_2 & nMask;
nRet <<= 8;
rStm >> nMask;
nRet |= nMask;
}
else if( nMask & LEN_4 )
{
nRet = ~LEN_4 & nMask;
nRet <<= 8;
rStm >> nMask;
nRet |= nMask;
nRet <<= 16;
sal_uInt16 n;
rStm >> n;
nRet |= n;
}
else if( nMask & LEN_5 )
{
if( nMask & 0x0F )
{
rStm.SetError( SVSTREAM_FILEFORMAT_ERROR );
DBG_ERROR( "format error" );
}
rStm >> nRet;
}
else
{
rStm.SetError( SVSTREAM_FILEFORMAT_ERROR );
DBG_ERROR( "format error" );
}
return nRet;
}
//=========================================================================
void SvPersistStream::WriteCompressed
(
SvStream & rStm,/* Aus diesem Stream werden die komprimierten Daten
gelesen */
sal_uInt32 nVal /* Dieser Wert wird komprimiert geschrieben */
)
/* [Beschreibung]
Das "ubergebene Wort wird komprimiert und in den Stream
geschrieben. Folgendermassen wir komprimiert.
nVal < 0x80 => 0x80 + nVal ist 1 Byte gross.
nVal < 0x4000 => 0x4000 + nVal ist 2 Byte gross.
nVal < 0x20000000 => 0x20000000 + nVal ist 4 Byte gross.
nVal > 0x1FFFFFFF => 0x1000000000+ nVal ist 5 Byte gross.
[Querverweise]
<SvPersistStream::ReadCompressed>
*/
{
#ifdef STOR_NO_OPTIMIZE
if( nVal < 0x80 )
rStm << (sal_uInt8)(LEN_1 | nVal);
else if( nVal < 0x4000 )
{
rStm << (sal_uInt8)(LEN_2 | (nVal >> 8));
rStm << (sal_uInt8)nVal;
}
else if( nVal < 0x20000000 )
{
// hoechstes sal_uInt8
rStm << (sal_uInt8)(LEN_4 | (nVal >> 24));
// 2. hoechstes sal_uInt8
rStm << (sal_uInt8)(nVal >> 16);
rStm << (sal_uInt16)(nVal);
}
else
#endif
{
rStm << (sal_uInt8)LEN_5;
rStm << nVal;
}
}
//=========================================================================
sal_uInt32 SvPersistStream::WriteDummyLen()
/* [Beschreibung]
Die Methode schreibt 4 Byte in den Stream und gibt die Streamposition
zur"uck.
[R"uckgabewert]
sal_uInt32 Die Position hinter der L"angenangabe wird zur"uckgegeben.
[Beispiel]
sal_uInt32 nObjPos = rStm.WriteDummyLen();
...
// Daten schreiben
...
rStm.WriteLen( nObjPos );
[Querverweise]
<SvPersistStream::ReadLen>, <SvPersistStream::WriteLen>
*/
{
#ifdef DBG_UTIL
sal_uInt32 nPos = Tell();
#endif
sal_uInt32 n0 = 0;
*this << n0; // wegen Sun sp
// keine Assertion bei Streamfehler
DBG_ASSERT( GetError() != SVSTREAM_OK
|| (sizeof( sal_uInt32 ) == Tell() -nPos),
"keine 4-Byte fuer Langenangabe" );
return Tell();
}
//=========================================================================
void SvPersistStream::WriteLen
(
sal_uInt32 nObjPos /* die Position + 4, an der die L"ange geschrieben
wird. */
)
/* [Beschreibung]
Die Methode schreibt die Differenz zwischen der aktuellen und
nObjPos als sal_uInt32 an die Position nObjPos -4 im Stream. Danach
wird der Stream wieder auf die alte Position gestellt.
[Beispiel]
Die Differenz enth"alt nicht die L"angenangabe.
sal_uInt32 nObjPos = rStm.WriteDummyLen();
...
// Daten schreiben
...
rStm.WriteLen( nObjPos );
// weitere Daten schreiben
[Querverweise]
<SvPersistStream::ReadLen>, <SvPersistStream::WriteDummyLen>
*/
{
sal_uInt32 nPos = Tell();
sal_uInt32 nLen = nPos - nObjPos;
// die Laenge mu� im stream 4-Byte betragen
Seek( nObjPos - sizeof( sal_uInt32 ) );
// Laenge schreiben
*this << nLen;
Seek( nPos );
}
//=========================================================================
sal_uInt32 SvPersistStream::ReadLen
(
sal_uInt32 * pTestPos /* Die Position des Streams, nach dem Lesen der
L"ange, wird zur"uckgegeben. Es darf auch NULL
"ubergeben werden. */
)
/* [Beschreibung]
Liest die L"ange die vorher mit <SvPersistStream::WriteDummyLen>
und <SvPersistStream::WriteLen> geschrieben wurde.
*/
{
sal_uInt32 nLen;
*this >> nLen;
if( pTestPos )
*pTestPos = Tell();
return nLen;
}
//=========================================================================
// Dateirormat abw"arts kompatibel
#ifdef STOR_NO_OPTIMIZE
#define P_VER (sal_uInt8)0x00
#else
#define P_VER (sal_uInt8)0x01
#endif
#define P_VER_MASK (sal_uInt8)0x0F
#define P_ID_0 (sal_uInt8)0x80
#define P_OBJ (sal_uInt8)0x40
#define P_DBGUTIL (sal_uInt8)0x20
#define P_ID (sal_uInt8)0x10
#ifdef STOR_NO_OPTIMIZE
#define P_STD P_DBGUTIL
#else
#define P_STD 0
#endif
static void WriteId
(
SvStream & rStm,
sal_uInt8 nHdr,
sal_uInt32 nId,
sal_uInt16 nClassId
)
{
#ifdef STOR_NO_OPTIMIZE
nHdr |= P_ID;
#endif
nHdr |= P_VER;
if( nHdr & P_ID )
{
if( (nHdr & P_OBJ) || nId != 0 )
{ // Id nur bei Zeiger, oder DBGUTIL
rStm << (sal_uInt8)(nHdr);
SvPersistStream::WriteCompressed( rStm, nId );
}
else
{ // NULL Pointer
rStm << (sal_uInt8)(nHdr | P_ID_0);
return;
}
}
else
rStm << nHdr;
if( (nHdr & P_DBGUTIL) || (nHdr & P_OBJ) )
// Objekte haben immer eine Klasse,
// Pointer nur bei DBG_UTIL und != NULL
SvPersistStream::WriteCompressed( rStm, nClassId );
}
//=========================================================================
static void ReadId
(
SvStream & rStm,
sal_uInt8 & nHdr,
sal_uInt32 & nId,
sal_uInt16 & nClassId
)
{
nClassId = 0;
rStm >> nHdr;
if( nHdr & P_ID_0 )
nId = 0;
else
{
if( (nHdr & P_VER_MASK) == 0 )
{
if( (nHdr & P_DBGUTIL) || !(nHdr & P_OBJ) )
nId = SvPersistStream::ReadCompressed( rStm );
else
nId = 0;
}
else if( nHdr & P_ID )
nId = SvPersistStream::ReadCompressed( rStm );
if( (nHdr & P_DBGUTIL) || (nHdr & P_OBJ) )
nClassId = (sal_uInt16)SvPersistStream::ReadCompressed( rStm );
}
}
//=========================================================================
void SvPersistStream::WriteObj
(
sal_uInt8 nHdr,
SvPersistBase * pObj
)
{
#ifdef STOR_NO_OPTIMIZE
sal_uInt32 nObjPos = 0;
if( nHdr & P_DBGUTIL )
// Position fuer Laenge merken
nObjPos = WriteDummyLen();
#endif
pObj->Save( *this );
#ifdef STOR_NO_OPTIMIZE
if( nHdr & P_DBGUTIL )
WriteLen( nObjPos );
#endif
}
//=========================================================================
SvPersistStream& SvPersistStream::WritePointer
(
SvPersistBase * pObj
)
{
sal_uInt8 nP = P_STD;
if( pObj )
{
sal_uIntPtr nId = GetIndex( pObj );
if( nId )
nP |= P_ID;
else
{
nId = aPUIdx.Insert( pObj );
aPTable.Insert( (sal_uIntPtr)pObj, (void *)nId );
nP |= P_OBJ;
}
WriteId( *this, nP, nId, pObj->GetClassId() );
if( nP & P_OBJ )
WriteObj( nP, pObj );
}
else
{ // NULL Pointer
WriteId( *this, nP | P_ID, 0, 0 );
}
return *this;
}
//=========================================================================
sal_uInt32 SvPersistStream::ReadObj
(
SvPersistBase * & rpObj,
sal_Bool bRegister
)
{
sal_uInt8 nHdr;
sal_uInt32 nId = 0;
sal_uInt16 nClassId;
rpObj = NULL; // Spezifikation: Im Fehlerfall 0.
ReadId( *this, nHdr, nId, nClassId );
// reine Versionsnummer durch maskieren
if( P_VER < (nHdr & P_VER_MASK) )
{
SetError( SVSTREAM_FILEFORMAT_ERROR );
DBG_ERROR( "false version" );
}
if( !(nHdr & P_ID_0) && GetError() == SVSTREAM_OK )
{
if( P_OBJ & nHdr )
{ // read object, nId nur bei P_DBGUTIL gesetzt
DBG_ASSERT( !(nHdr & P_DBGUTIL) || NULL == aPUIdx.Get( nId ),
"object already exist" );
SvCreateInstancePersist pFunc = rClassMgr.Get( nClassId );
sal_uInt32 nObjLen(0), nObjPos(0);
if( nHdr & P_DBGUTIL )
nObjLen = ReadLen( &nObjPos );
if( !pFunc )
{
#ifdef DBG_UTIL
ByteString aStr( "no class with id: " );
aStr += ByteString::CreateFromInt32( nClassId );
aStr += " registered";
DBG_WARNING( aStr.GetBuffer() );
#endif
SetError( ERRCODE_IO_NOFACTORY );
return 0;
}
pFunc( &rpObj );
// Sichern
rpObj->AddRef();
if( bRegister )
{
// unbedingt erst in Tabelle eintragen
sal_uIntPtr nNewId = aPUIdx.Insert( rpObj );
// um den gleichen Zustand, wie nach dem Speichern herzustellen
aPTable.Insert( (sal_uIntPtr)rpObj, (void *)nNewId );
DBG_ASSERT( !(nHdr & P_DBGUTIL) || nId == nNewId,
"read write id conflict: not the same" );
}
// und dann Laden
rpObj->Load( *this );
#ifdef DBG_UTIL
if( nObjLen + nObjPos != Tell() )
{
ByteString aStr( "false object len: read = " );
aStr += ByteString::CreateFromInt32( (long)(Tell() - nObjPos) );
aStr += ", should = ";
aStr += ByteString::CreateFromInt32( nObjLen );
DBG_ERROR( aStr.GetBuffer() );
}
#endif
rpObj->RestoreNoDelete();
rpObj->ReleaseRef();
}
else
{
rpObj = GetObject( nId );
DBG_ASSERT( rpObj != NULL, "object does not exist" );
DBG_ASSERT( rpObj->GetClassId() == nClassId, "class mismatch" );
}
}
return nId;
}
//=========================================================================
SvPersistStream& SvPersistStream::ReadPointer
(
SvPersistBase * & rpObj
)
{
ReadObj( rpObj, sal_True );
return *this;
}
//=========================================================================
SvPersistStream& operator <<
(
SvPersistStream & rStm,
SvPersistBase * pObj
)
{
return rStm.WritePointer( pObj );
}
//=========================================================================
SvPersistStream& operator >>
(
SvPersistStream & rStm,
SvPersistBase * & rpObj
)
{
return rStm.ReadPointer( rpObj );
}
//=========================================================================
SvStream& operator <<
(
SvStream & rStm,
SvPersistStream & rThis
)
{
SvStream * pOldStm = rThis.GetStream();
rThis.SetStream( &rStm );
sal_uInt8 bTmp = 0;
rThis << bTmp; // Version
sal_uInt32 nCount = (sal_uInt32)rThis.aPUIdx.Count();
rThis << nCount;
SvPersistBase * pEle = rThis.aPUIdx.First();
for( sal_uInt32 i = 0; i < nCount; i++ )
{
sal_uInt8 nP = P_OBJ | P_ID | P_STD;
WriteId( rThis, nP, rThis.aPUIdx.GetCurIndex(),
pEle->GetClassId() );
rThis.WriteObj( nP, pEle );
pEle = rThis.aPUIdx.Next();
}
rThis.SetStream( pOldStm );
return rStm;
}
//=========================================================================
SvStream& operator >>
(
SvStream & rStm,
SvPersistStream & rThis
)
{
SvStream * pOldStm = rThis.GetStream();
rThis.SetStream( &rStm );
sal_uInt8 nVers;
rThis >> nVers; // Version
if( 0 == nVers )
{
sal_uInt32 nCount = 0;
rThis >> nCount;
for( sal_uInt32 i = 0; i < nCount; i++ )
{
SvPersistBase * pEle;
// Lesen, ohne in die Tabellen einzutragen
sal_uInt32 nId = rThis.ReadObj( pEle, sal_False );
if( rThis.GetError() )
break;
// Die Id eines Objektes wird nie modifiziert
rThis.aPUIdx.Insert( nId, pEle );
rThis.aPTable.Insert( (sal_uIntPtr)pEle, (void *)nId );
}
}
else
rThis.SetError( SVSTREAM_FILEFORMAT_ERROR );
rThis.SetStream( pOldStm );
return rStm;
}
//=========================================================================
sal_uIntPtr SvPersistStream::InsertObj( SvPersistBase * pObj )
{
sal_uIntPtr nId = aPUIdx.Insert( pObj );
aPTable.Insert( (sal_uIntPtr)pObj, (void *)nId );
return nId;
}
//=========================================================================
sal_uIntPtr SvPersistStream::RemoveObj( SvPersistBase * pObj )
{
sal_uIntPtr nIdx = GetIndex( pObj );
aPUIdx.Remove( nIdx );
aPTable.Remove( (sal_uIntPtr)pObj );
return nIdx;
}