/************************************************************** * * 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 #include #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 ). */ { 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 ). 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 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 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] */ { 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 . [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] */ { #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] , */ { #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] , */ { 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 und 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; }