1 /************************************************************** 2 * 3 * Licensed to the Apache Software Foundation (ASF) under one 4 * or more contributor license agreements. See the NOTICE file 5 * distributed with this work for additional information 6 * regarding copyright ownership. The ASF licenses this file 7 * to you under the Apache License, Version 2.0 (the 8 * "License"); you may not use this file except in compliance 9 * with the License. You may obtain a copy of the License at 10 * 11 * http://www.apache.org/licenses/LICENSE-2.0 12 * 13 * Unless required by applicable law or agreed to in writing, 14 * software distributed under the License is distributed on an 15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16 * KIND, either express or implied. See the License for the 17 * specific language governing permissions and limitations 18 * under the License. 19 * 20 *************************************************************/ 21 22 23 24 // MARKER(update_precomp.py): autogen include statement, do not remove 25 #include "precompiled_sot.hxx" 26 27 #include <string.h> // memcpy() 28 29 #include "sot/stg.hxx" 30 #include "stgelem.hxx" 31 #include "stgcache.hxx" 32 #include "stgstrms.hxx" 33 #include "stgdir.hxx" 34 #include "stgio.hxx" 35 36 37 //////////////////////////// class StgDirEntry ///////////////////////////// 38 39 // This class holds the dir entry data and maintains dirty flags for both 40 // the entry and the data. 41 42 // Transacted mode for streams: On the first write, a temp stream pTmpStrm 43 // is created and operated on. A commit moves pTmpStrm to pCurStrm, which 44 // is used for subsequent reads. A new write creates a new copy of pTmpStrm 45 // based on pCurStrm. Reverting throws away pTmpStrm. 46 // Transacted mode for storages: A copy of the dir ents is kept in aSave. 47 // Committing means copying aEntry to aSave. Reverting means to copy aSave 48 // to aEntry, delete newly created entries and to reactivate removed entries. 49 50 // Problem der Implementation: Keine Hierarchischen commits. Daher nur 51 // insgesamt transaktionsorientert oder direkt. 52 53 StgDirEntry::StgDirEntry( const void* pBuffer, sal_uInt32 nBufferLen, sal_Bool * pbOk ) : StgAvlNode() 54 { 55 *pbOk = aEntry.Load( pBuffer, nBufferLen ); 56 57 InitMembers(); 58 } 59 60 StgDirEntry::StgDirEntry( const StgEntry& r ) : StgAvlNode(), aEntry( r ) 61 { 62 InitMembers(); 63 } 64 65 // Helper for all ctors 66 67 void StgDirEntry::InitMembers() 68 { 69 aSave = aEntry; 70 pUp = 71 pDown = NULL; 72 ppRoot = NULL; 73 pStgStrm = NULL; 74 pCurStrm = 75 pTmpStrm = NULL; 76 nPos = 77 nEntry = 78 nRefCnt = 0; 79 nMode = STREAM_READ; 80 bDirect = sal_True; 81 bInvalid = 82 bCreated = 83 bRenamed = 84 bRemoved = 85 bTemp = 86 bDirty = 87 bZombie = sal_False; 88 } 89 90 StgDirEntry::~StgDirEntry() 91 { 92 Close(); 93 delete pCurStrm; 94 delete pStgStrm; 95 delete pDown; 96 } 97 98 // Comparison function 99 100 short StgDirEntry::Compare( const StgAvlNode* p ) const 101 { 102 short nResult = -1; 103 if ( p ) 104 { 105 const StgDirEntry* pEntry = (const StgDirEntry*) p; 106 nResult = aEntry.Compare( pEntry->aEntry ); 107 } 108 return nResult; 109 } 110 111 // Enumerate the entry numbers. 112 // n is incremented to show the total # of entries. 113 // These number are later used as page numbers when storing 114 // the TOC tree into the TOC stream. Remember that aSave is 115 // stored, not aEntry. 116 117 void StgDirEntry::Enum( sal_Int32& n ) 118 { 119 sal_Int32 nLeft = STG_FREE, nRight = STG_FREE, nDown = STG_FREE; 120 nEntry = n++; 121 if( pLeft ) 122 { 123 ((StgDirEntry*) pLeft)->Enum( n ); nLeft = ((StgDirEntry*) pLeft)->nEntry; 124 } 125 if( pRight ) 126 { 127 ((StgDirEntry*) pRight)->Enum( n ); nRight = ((StgDirEntry*) pRight)->nEntry; 128 } 129 if( pDown ) 130 { 131 pDown->Enum( n ); nDown = pDown->nEntry; 132 } 133 aSave.SetLeaf( STG_LEFT, nLeft ); 134 aSave.SetLeaf( STG_RIGHT, nRight ); 135 aSave.SetLeaf( STG_CHILD, nDown ); 136 } 137 138 // Delete all temporary entries before writing the TOC stream. 139 // Until now Deltem is never called with bForce True 140 141 void StgDirEntry::DelTemp( sal_Bool bForce ) 142 { 143 if( pLeft ) 144 ((StgDirEntry*) pLeft)->DelTemp( sal_False ); 145 if( pRight ) 146 ((StgDirEntry*) pRight)->DelTemp( sal_False ); 147 if( pDown ) 148 { 149 // If the storage is dead, of course all elements are dead, too 150 if( bInvalid && aEntry.GetType() == STG_STORAGE ) 151 bForce = sal_True; 152 pDown->DelTemp( bForce ); 153 } 154 if( ( bForce || bInvalid ) 155 && ( aEntry.GetType() != STG_ROOT ) /* && ( nRefCnt <= 1 ) */ ) 156 { 157 Close(); 158 if( pUp ) 159 { 160 // this deletes the element if refcnt == 0! 161 sal_Bool bDel = nRefCnt == 0; 162 StgAvlNode::Remove( (StgAvlNode**) &pUp->pDown, this, bDel ); 163 if( !bDel ) 164 { 165 pLeft = pRight = pDown = 0; 166 bInvalid = bZombie = sal_True; 167 } 168 } 169 } 170 } 171 172 // Save the tree into the given dir stream 173 174 sal_Bool StgDirEntry::Store( StgDirStrm& rStrm ) 175 { 176 void* pEntry = rStrm.GetEntry( nEntry, sal_True ); 177 if( !pEntry ) 178 return sal_False; 179 // Do not store the current (maybe not commited) entry 180 aSave.Store( pEntry ); 181 if( pLeft ) 182 if( !((StgDirEntry*) pLeft)->Store( rStrm ) ) 183 return sal_False; 184 if( pRight ) 185 if( !((StgDirEntry*) pRight)->Store( rStrm ) ) 186 return sal_False; 187 if( pDown ) 188 if( !pDown->Store( rStrm ) ) 189 return sal_False; 190 return sal_True; 191 } 192 193 sal_Bool StgDirEntry::StoreStream( StgIo& rIo ) 194 { 195 if( aEntry.GetType() == STG_STREAM || aEntry.GetType() == STG_ROOT ) 196 { 197 if( bInvalid ) 198 { 199 // Delete the stream if needed 200 if( !pStgStrm ) 201 { 202 OpenStream( rIo ); 203 delete pStgStrm, pStgStrm = NULL; 204 } 205 else 206 pStgStrm->SetSize( 0 ); 207 } 208 // or write the data stream 209 else if( !Tmp2Strm() ) 210 return sal_False; 211 } 212 return sal_True; 213 } 214 215 // Save all dirty streams 216 217 sal_Bool StgDirEntry::StoreStreams( StgIo& rIo ) 218 { 219 if( !StoreStream( rIo ) ) 220 return sal_False; 221 if( pLeft ) 222 if( !((StgDirEntry*) pLeft)->StoreStreams( rIo ) ) 223 return sal_False; 224 if( pRight ) 225 if( !((StgDirEntry*) pRight)->StoreStreams( rIo ) ) 226 return sal_False; 227 if( pDown ) 228 if( !pDown->StoreStreams( rIo ) ) 229 return sal_False; 230 return sal_True; 231 } 232 233 // Revert all directory entries after failure to write the TOC stream 234 235 void StgDirEntry::RevertAll() 236 { 237 aEntry = aSave; 238 if( pLeft ) 239 ((StgDirEntry*) pLeft)->RevertAll(); 240 if( pRight ) 241 ((StgDirEntry*) pRight)->RevertAll(); 242 if( pDown ) 243 pDown->RevertAll(); 244 } 245 246 // Look if any element of the tree is dirty 247 248 sal_Bool StgDirEntry::IsDirty() 249 { 250 if( bDirty || bInvalid ) 251 return sal_True; 252 if( pLeft && ((StgDirEntry*) pLeft)->IsDirty() ) 253 return sal_True; 254 if( pRight && ((StgDirEntry*) pRight)->IsDirty() ) 255 return sal_True; 256 if( pDown && pDown->IsDirty() ) 257 return sal_True; 258 return sal_False; 259 } 260 261 // Set up a stream. 262 263 void StgDirEntry::OpenStream( StgIo& rIo, sal_Bool bForceBig ) 264 { 265 sal_Int32 nThreshold = (sal_uInt16) rIo.aHdr.GetThreshold(); 266 delete pStgStrm; 267 if( !bForceBig && aEntry.GetSize() < nThreshold ) 268 pStgStrm = new StgSmallStrm( rIo, *this ); 269 else 270 pStgStrm = new StgDataStrm( rIo, *this ); 271 if( bInvalid && aEntry.GetSize() ) 272 { 273 // This entry has invalid data, so delete that data 274 SetSize( 0L ); 275 // bRemoved = bInvalid = sal_False; 276 } 277 nPos = 0; 278 } 279 280 // Close the open stream without committing. If the entry is marked as 281 // temporary, delete it. 282 // Do not delete pCurStrm here! 283 // (TLX:??? Zumindest pStgStrm muss deleted werden.) 284 285 void StgDirEntry::Close() 286 { 287 delete pTmpStrm; 288 pTmpStrm = NULL; 289 // nRefCnt = 0; 290 bInvalid = bTemp; 291 } 292 293 // Get the current stream size 294 295 sal_Int32 StgDirEntry::GetSize() 296 { 297 sal_Int32 n; 298 if( pTmpStrm ) 299 n = pTmpStrm->GetSize(); 300 else if( pCurStrm ) 301 n = pCurStrm->GetSize(); 302 else n = aEntry.GetSize(); 303 return n; 304 } 305 306 // Set the stream size. This means also creating a temp stream. 307 308 sal_Bool StgDirEntry::SetSize( sal_Int32 nNewSize ) 309 { 310 if ( 311 !( nMode & STREAM_WRITE ) || 312 (!bDirect && !pTmpStrm && !Strm2Tmp()) 313 ) 314 { 315 return sal_False; 316 } 317 318 if( nNewSize < nPos ) 319 nPos = nNewSize; 320 if( pTmpStrm ) 321 { 322 pTmpStrm->SetSize( nNewSize ); 323 pStgStrm->GetIo().SetError( pTmpStrm->GetError() ); 324 return sal_Bool( pTmpStrm->GetError() == SVSTREAM_OK ); 325 } 326 else 327 { 328 OSL_ENSURE( pStgStrm, "The pointer may not be NULL!" ); 329 if ( !pStgStrm ) 330 return sal_False; 331 332 sal_Bool bRes = sal_False; 333 StgIo& rIo = pStgStrm->GetIo(); 334 sal_Int32 nThreshold = rIo.aHdr.GetThreshold(); 335 // ensure the correct storage stream! 336 StgStrm* pOld = NULL; 337 sal_uInt16 nOldSize = 0; 338 if( nNewSize >= nThreshold && pStgStrm->IsSmallStrm() ) 339 { 340 pOld = pStgStrm; 341 nOldSize = (sal_uInt16) pOld->GetSize(); 342 pStgStrm = new StgDataStrm( rIo, STG_EOF, 0 ); 343 } 344 else if( nNewSize < nThreshold && !pStgStrm->IsSmallStrm() ) 345 { 346 pOld = pStgStrm; 347 nOldSize = (sal_uInt16) nNewSize; 348 pStgStrm = new StgSmallStrm( rIo, STG_EOF, 0 ); 349 } 350 // now set the new size 351 if( pStgStrm->SetSize( nNewSize ) ) 352 { 353 // did we create a new stream? 354 if( pOld ) 355 { 356 // if so, we probably need to copy the old data 357 if( nOldSize ) 358 { 359 void* pBuf = new sal_uInt8[ nOldSize ]; 360 pOld->Pos2Page( 0L ); 361 pStgStrm->Pos2Page( 0L ); 362 if( pOld->Read( pBuf, nOldSize ) 363 && pStgStrm->Write( pBuf, nOldSize ) ) 364 bRes = sal_True; 365 delete[] static_cast<sal_uInt8*>(pBuf); 366 } 367 else 368 bRes = sal_True; 369 if( bRes ) 370 { 371 pOld->SetSize( 0 ); 372 delete pOld; 373 pStgStrm->Pos2Page( nPos ); 374 pStgStrm->SetEntry( *this ); 375 } 376 else 377 { 378 pStgStrm->SetSize( 0 ); 379 delete pStgStrm; 380 pStgStrm = pOld; 381 } 382 } 383 else 384 { 385 pStgStrm->Pos2Page( nPos ); 386 bRes = sal_True; 387 } 388 } 389 return bRes; 390 } 391 } 392 393 // Seek. On negative values, seek to EOF. 394 395 sal_Int32 StgDirEntry::Seek( sal_Int32 nNew ) 396 { 397 if( pTmpStrm ) 398 { 399 if( nNew < 0 ) 400 nNew = pTmpStrm->GetSize(); 401 nNew = pTmpStrm->Seek( nNew ); 402 } 403 else if( pCurStrm ) 404 { 405 if( nNew < 0 ) 406 nNew = pCurStrm->GetSize(); 407 nNew = pCurStrm->Seek( nNew ); 408 } 409 else 410 { 411 OSL_ENSURE( pStgStrm, "The pointer may not be NULL!" ); 412 if ( !pStgStrm ) 413 return nPos; 414 415 sal_Int32 nSize = aEntry.GetSize(); 416 417 if( nNew < 0 ) 418 nNew = nSize; 419 420 // try to enlarge, the readonly streams should not allow this 421 if( nNew > nSize ) 422 { 423 if ( !( nMode & STREAM_WRITE ) || !SetSize( nNew ) ) 424 { 425 OSL_ENSURE( nMode & STREAM_WRITE, "Trying to resize readonly stream by seeking, could be a wrong offset!" ); 426 return nPos; 427 } 428 else 429 return Seek( nNew ); 430 } 431 pStgStrm->Pos2Page( nNew ); 432 nNew = pStgStrm->GetPos(); 433 } 434 435 return nPos = nNew; 436 } 437 438 // Read 439 440 sal_Int32 StgDirEntry::Read( void* p, sal_Int32 nLen ) 441 { 442 if( nLen <= 0 ) 443 return 0; 444 if( pTmpStrm ) 445 nLen = pTmpStrm->Read( p, nLen ); 446 else if( pCurStrm ) 447 nLen = pCurStrm->Read( p, nLen ); 448 else 449 { 450 OSL_ENSURE( pStgStrm, "The pointer may not be NULL!" ); 451 if ( !pStgStrm ) 452 return 0; 453 454 nLen = pStgStrm->Read( p, nLen ); 455 } 456 457 nPos += nLen; 458 return nLen; 459 } 460 461 // Write 462 463 sal_Int32 StgDirEntry::Write( const void* p, sal_Int32 nLen ) 464 { 465 if( nLen <= 0 || !( nMode & STREAM_WRITE ) ) 466 return 0; 467 468 // Was this stream committed internally and reopened in direct mode? 469 if( bDirect && ( pCurStrm || pTmpStrm ) && !Tmp2Strm() ) 470 return 0; 471 // Is this stream opened in transacted mode? Do we have to make a copy? 472 if( !bDirect && !pTmpStrm && !Strm2Tmp() ) 473 return 0; 474 475 OSL_ENSURE( pStgStrm, "The pointer may not be NULL!" ); 476 if ( !pStgStrm ) 477 return 0; 478 479 if( pTmpStrm ) 480 { 481 nLen = pTmpStrm->Write( p, nLen ); 482 pStgStrm->GetIo().SetError( pTmpStrm->GetError() ); 483 } 484 else 485 { 486 sal_Int32 nNew = nPos + nLen; 487 if( nNew > pStgStrm->GetSize() ) 488 { 489 if( !SetSize( nNew ) ) 490 return 0L; 491 pStgStrm->Pos2Page( nPos ); 492 } 493 nLen = pStgStrm->Write( p, nLen ); 494 } 495 nPos += nLen; 496 return nLen; 497 } 498 499 // Copy the data of one entry into another entry. 500 501 void StgDirEntry::Copy( StgDirEntry& rDest ) 502 { 503 sal_Int32 n = GetSize(); 504 if( rDest.SetSize( n ) && n ) 505 { 506 sal_uInt8 aTempBytes[ 4096 ]; 507 void* p = static_cast<void*>( aTempBytes ); 508 Seek( 0L ); 509 rDest.Seek( 0L ); 510 while( n ) 511 { 512 sal_Int32 nn = n; 513 if( nn > 4096 ) 514 nn = 4096; 515 if( Read( p, nn ) != nn ) 516 break; 517 if( rDest.Write( p, nn ) != nn ) 518 break; 519 n -= nn; 520 } 521 } 522 } 523 524 void StgDirEntry::Copy( BaseStorageStream& rDest ) 525 { 526 sal_Int32 n = GetSize(); 527 if( rDest.SetSize( n ) && n ) 528 { 529 sal_uLong Pos = rDest.Tell(); 530 sal_uInt8 aTempBytes[ 4096 ]; 531 void* p = static_cast<void*>( aTempBytes ); 532 Seek( 0L ); 533 rDest.Seek( 0L ); 534 while( n ) 535 { 536 sal_Int32 nn = n; 537 if( nn > 4096 ) 538 nn = 4096; 539 if( Read( p, nn ) != nn ) 540 break; 541 if( sal::static_int_cast<sal_Int32>(rDest.Write( p, nn )) != nn ) 542 break; 543 n -= nn; 544 } 545 rDest.Seek( Pos ); // ?! Seems to be undocumented ! 546 } 547 } 548 549 // Commit this entry 550 551 sal_Bool StgDirEntry::Commit() 552 { 553 // OSL_ENSURE( nMode & STREAM_WRITE, "Trying to commit readonly stream!" ); 554 555 aSave = aEntry; 556 sal_Bool bRes = sal_True; 557 if( aEntry.GetType() == STG_STREAM ) 558 { 559 if( pTmpStrm ) 560 delete pCurStrm, pCurStrm = pTmpStrm, pTmpStrm = NULL; 561 if( bRemoved ) 562 // Delete the stream if needed 563 if( pStgStrm ) 564 pStgStrm->SetSize( 0 ); 565 } 566 else if( aEntry.GetType() == STG_STORAGE && bDirect && bRes ) 567 { 568 StgIterator aIter( *this ); 569 for( StgDirEntry* p = aIter.First(); p && bRes; p = aIter.Next() ) 570 bRes = p->Commit(); 571 } 572 return bRes; 573 } 574 575 // Revert the entry 576 577 sal_Bool StgDirEntry::Revert() 578 { 579 aEntry = aSave; 580 switch( aEntry.GetType() ) 581 { 582 case STG_STREAM: 583 if( pCurStrm ) 584 delete pTmpStrm, pTmpStrm = pCurStrm, pCurStrm = NULL; 585 break; 586 case STG_STORAGE: 587 { 588 sal_Bool bSomeRenamed = sal_False; 589 StgIterator aOIter( *this ); 590 StgDirEntry* op = aOIter.First(); 591 while( op ) 592 { 593 op->aEntry = op->aSave; 594 op->bDirty = sal_False; 595 bSomeRenamed = sal_Bool( bSomeRenamed | op->bRenamed ); 596 // Remove any new entries 597 if( op->bCreated ) 598 { 599 op->bCreated = sal_False; 600 op->Close(); 601 op->bInvalid = sal_True; 602 } 603 // Reactivate any removed entries 604 else if( op->bRemoved ) 605 op->bRemoved = op->bInvalid = op->bTemp = sal_False; 606 op = aOIter.Next(); 607 } 608 // Resort all renamed entries 609 if( bSomeRenamed ) 610 { 611 StgIterator aIter( *this ); 612 StgDirEntry* p = aIter.First(); 613 while( p ) 614 { 615 if( p->bRenamed ) 616 { 617 StgAvlNode::Move 618 ( (StgAvlNode**) &p->pUp->pDown, 619 (StgAvlNode**) &p->pUp->pDown, p ); 620 p->bRenamed = sal_False; 621 } 622 p = aIter.Next(); 623 } 624 } 625 DelTemp( sal_False ); 626 break; 627 } 628 case STG_EMPTY: 629 case STG_LOCKBYTES: 630 case STG_PROPERTY: 631 case STG_ROOT: 632 break; 633 } 634 return sal_True; 635 } 636 637 // Copy the stg stream to the temp stream 638 639 sal_Bool StgDirEntry::Strm2Tmp() 640 { 641 if( !pTmpStrm ) 642 { 643 sal_uLong n = 0; 644 if( pCurStrm ) 645 { 646 // It was already commited once 647 pTmpStrm = new StgTmpStrm; 648 if( pTmpStrm->GetError() == SVSTREAM_OK && pTmpStrm->Copy( *pCurStrm ) ) 649 return sal_True; 650 n = 1; // indicates error 651 } 652 else 653 { 654 n = aEntry.GetSize(); 655 pTmpStrm = new StgTmpStrm( n ); 656 if( pTmpStrm->GetError() == SVSTREAM_OK ) 657 { 658 if( n ) 659 { 660 OSL_ENSURE( pStgStrm, "The pointer may not be NULL!" ); 661 if ( !pStgStrm ) 662 return sal_False; 663 664 sal_uInt8 aTempBytes[ 4096 ]; 665 void* p = static_cast<void*>( aTempBytes ); 666 pStgStrm->Pos2Page( 0L ); 667 while( n ) 668 { 669 sal_uLong nn = n; 670 if( nn > 4096 ) 671 nn = 4096; 672 if( (sal_uLong) pStgStrm->Read( p, nn ) != nn ) 673 break; 674 if( pTmpStrm->Write( p, nn ) != nn ) 675 break; 676 n -= nn; 677 } 678 pStgStrm->Pos2Page( nPos ); 679 pTmpStrm->Seek( nPos ); 680 } 681 } 682 else 683 n = 1; 684 } 685 686 if( n ) 687 { 688 OSL_ENSURE( pStgStrm, "The pointer may not be NULL!" ); 689 if ( pStgStrm ) 690 pStgStrm->GetIo().SetError( pTmpStrm->GetError() ); 691 692 delete pTmpStrm; 693 pTmpStrm = NULL; 694 return sal_False; 695 } 696 } 697 return sal_True; 698 } 699 700 // Copy the temp stream to the stg stream during the final commit 701 702 sal_Bool StgDirEntry::Tmp2Strm() 703 { 704 // We did commit once, but have not written since then 705 if( !pTmpStrm ) 706 pTmpStrm = pCurStrm, pCurStrm = NULL; 707 if( pTmpStrm ) 708 { 709 OSL_ENSURE( pStgStrm, "The pointer may not be NULL!" ); 710 if ( !pStgStrm ) 711 return sal_False; 712 sal_uLong n = pTmpStrm->GetSize(); 713 StgStrm* pNewStrm; 714 StgIo& rIo = pStgStrm->GetIo(); 715 sal_uLong nThreshold = (sal_uLong) rIo.aHdr.GetThreshold(); 716 if( n < nThreshold ) 717 pNewStrm = new StgSmallStrm( rIo, STG_EOF, 0 ); 718 else 719 pNewStrm = new StgDataStrm( rIo, STG_EOF, 0 ); 720 if( pNewStrm->SetSize( n ) ) 721 { 722 sal_uInt8 p[ 4096 ]; 723 pTmpStrm->Seek( 0L ); 724 while( n ) 725 { 726 sal_uLong nn = n; 727 if( nn > 4096 ) 728 nn = 4096; 729 if( pTmpStrm->Read( p, nn ) != nn ) 730 break; 731 if( (sal_uLong) pNewStrm->Write( p, nn ) != nn ) 732 break; 733 n -= nn; 734 } 735 if( n ) 736 { 737 pTmpStrm->Seek( nPos ); 738 pStgStrm->GetIo().SetError( pTmpStrm->GetError() ); 739 delete pNewStrm; 740 return sal_False; 741 } 742 else 743 { 744 pStgStrm->SetSize( 0L ); 745 delete pStgStrm; 746 pStgStrm = pNewStrm; 747 pNewStrm->SetEntry( *this ); 748 pNewStrm->Pos2Page( nPos ); 749 delete pTmpStrm; 750 delete pCurStrm; 751 pTmpStrm = pCurStrm = NULL; 752 aSave = aEntry; 753 } 754 } 755 } 756 return sal_True; 757 } 758 759 // Check if the given entry is contained in this entry 760 761 sal_Bool StgDirEntry::IsContained( StgDirEntry* pStg ) 762 { 763 if( aEntry.GetType() == STG_STORAGE ) 764 { 765 StgIterator aIter( *this ); 766 StgDirEntry* p = aIter.First(); 767 while( p ) 768 { 769 if( !p->aEntry.Compare( pStg->aEntry ) ) 770 return sal_False; 771 if( p->aEntry.GetType() == STG_STORAGE ) 772 if( !p->IsContained( pStg ) ) 773 return sal_False; 774 p = aIter.Next(); 775 } 776 } 777 return sal_True; 778 } 779 780 // Invalidate all open entries by setting the RefCount to 0. If the bDel 781 // flag is set, also set the invalid flag to indicate deletion during the 782 // next dir stream flush. 783 784 void StgDirEntry::Invalidate( sal_Bool bDel ) 785 { 786 // nRefCnt = 0; 787 if( bDel ) 788 bRemoved = bInvalid = sal_True; 789 switch( aEntry.GetType() ) 790 { 791 case STG_STORAGE: 792 case STG_ROOT: 793 { 794 StgIterator aIter( *this ); 795 for( StgDirEntry* p = aIter.First(); p; p = aIter.Next() ) 796 p->Invalidate( bDel ); 797 break; 798 } 799 default: 800 break; 801 } 802 } 803 804 ///////////////////////////// class StgDirStrm //////////////////////////// 805 806 // This specialized stream is the maintenance stream for the directory tree. 807 808 StgDirStrm::StgDirStrm( StgIo& r ) 809 : StgDataStrm( r, r.aHdr.GetTOCStart(), -1 ) 810 , pRoot( NULL ) 811 , nEntries( 0 ) 812 { 813 if( r.GetError() ) 814 return; 815 nEntries = nPageSize / STGENTRY_SIZE; 816 if( nStart == STG_EOF ) 817 { 818 StgEntry aRoot; 819 aRoot.Init(); 820 aRoot.SetName( String::CreateFromAscii( RTL_CONSTASCII_STRINGPARAM( "Root Entry" ) ) ); 821 aRoot.SetType( STG_ROOT ); 822 pRoot = new StgDirEntry( aRoot ); 823 pRoot->SetDirty(); 824 } 825 else 826 { 827 // temporarily use this instance as owner, so 828 // the TOC pages can be removed. 829 pEntry = (StgDirEntry*) this; // just for a bit pattern 830 SetupEntry(0, pRoot, nSize/STGENTRY_SIZE, 0); 831 rIo.Revert( pEntry ); 832 pEntry = NULL; 833 } 834 } 835 836 StgDirStrm::~StgDirStrm() 837 { 838 delete pRoot; 839 } 840 841 // Recursively parse the directory tree during reading the TOC stream 842 843 void StgDirStrm::SetupEntry ( 844 const sal_Int32 n, 845 StgDirEntry* pUpper, 846 const sal_Int32 nEntryCount, 847 const sal_Int32 nDepth) 848 { 849 if (nDepth > nEntryCount) 850 { 851 // Tree grew higher than there are different nodes. Looks like 852 // something is wrong with the file. Return now to avoid 853 // infinite recursion. 854 return; 855 } 856 else if (n>=nEntryCount || (n<0 && n!=STG_FREE)) 857 { 858 // n has an invalid value. Don't access the corresponding 859 // stream content. 860 return; 861 } 862 863 void* p = ( n == STG_FREE ) ? NULL : GetEntry( n ); 864 if( p ) 865 { 866 sal_Bool bOk(sal_False); 867 StgDirEntry* pCur = new StgDirEntry( p, STGENTRY_SIZE, &bOk ); 868 869 if( !bOk ) 870 { 871 delete pCur; 872 rIo.SetError( SVSTREAM_GENERALERROR ); 873 // an error occured 874 return; 875 } 876 877 // better it is 878 if( !pUpper ) 879 pCur->aEntry.SetType( STG_ROOT ); 880 881 sal_Int32 nLeft = pCur->aEntry.GetLeaf( STG_LEFT ); 882 sal_Int32 nRight = pCur->aEntry.GetLeaf( STG_RIGHT ); 883 // substorage? 884 sal_Int32 nLeaf = STG_FREE; 885 if( pCur->aEntry.GetType() == STG_STORAGE || pCur->aEntry.GetType() == STG_ROOT ) 886 { 887 nLeaf = pCur->aEntry.GetLeaf( STG_CHILD ); 888 if (nLeaf != STG_FREE && nLeaf == n) 889 { 890 delete pCur; 891 rIo.SetError( SVSTREAM_GENERALERROR ); 892 return; 893 } 894 } 895 896 if( nLeaf != 0 && nLeft != 0 && nRight != 0 ) 897 { 898 if( StgAvlNode::Insert 899 ( (StgAvlNode**) ( pUpper ? &pUpper->pDown : &pRoot ), pCur ) ) 900 { 901 pCur->pUp = pUpper; 902 pCur->ppRoot = &pRoot; 903 } 904 else 905 { 906 rIo.SetError( SVSTREAM_CANNOT_MAKE ); 907 delete pCur; pCur = NULL; 908 return; 909 } 910 SetupEntry( nLeft, pUpper, nEntryCount, nDepth+1); 911 SetupEntry( nRight, pUpper, nEntryCount, nDepth+1); 912 SetupEntry( nLeaf, pCur, nEntryCount, nDepth+1); 913 } 914 } 915 } 916 917 // Extend or shrink the directory stream. 918 919 sal_Bool StgDirStrm::SetSize( sal_Int32 nBytes ) 920 { 921 // Always allocate full pages 922 if ( nBytes < 0 ) 923 nBytes = 0; 924 925 nBytes = ( ( nBytes + nPageSize - 1 ) / nPageSize ) * nPageSize; 926 return StgStrm::SetSize( nBytes ); 927 } 928 929 // Save the TOC stream into a new substream after saving all data streams 930 931 sal_Bool StgDirStrm::Store() 932 { 933 if( !pRoot || !pRoot->IsDirty() ) 934 return sal_True; 935 if( !pRoot->StoreStreams( rIo ) ) 936 return sal_False; 937 // After writing all streams, the data FAT stream has changed, 938 // so we have to commit the root again 939 pRoot->Commit(); 940 // We want a completely new stream, so fake an empty stream 941 sal_Int32 nOldStart = nStart; // save for later deletion 942 sal_Int32 nOldSize = nSize; 943 nStart = nPage = STG_EOF; 944 nSize = nPos = 0; 945 nOffset = 0; 946 // Delete all temporary entries 947 pRoot->DelTemp( sal_False ); 948 // set the entry numbers 949 sal_Int32 n = 0; 950 pRoot->Enum( n ); 951 if( !SetSize( n * STGENTRY_SIZE ) ) 952 { 953 nStart = nOldStart; nSize = nOldSize; 954 pRoot->RevertAll(); 955 return sal_False; 956 } 957 // set up the cache elements for the new stream 958 if( !Copy( STG_FREE, nSize ) ) 959 { 960 pRoot->RevertAll(); 961 return sal_False; 962 } 963 // Write the data to the new stream 964 if( !pRoot->Store( *this ) ) 965 { 966 pRoot->RevertAll(); 967 return sal_False; 968 } 969 // fill any remaining entries with empty data 970 sal_Int32 ne = nSize / STGENTRY_SIZE; 971 StgEntry aEmpty; 972 aEmpty.Init(); 973 while( n < ne ) 974 { 975 void* p = GetEntry( n++, sal_True ); 976 if( !p ) 977 { 978 pRoot->RevertAll(); 979 return sal_False; 980 } 981 aEmpty.Store( p ); 982 } 983 // Now we can release the old stream 984 pFat->FreePages( nOldStart, sal_True ); 985 rIo.aHdr.SetTOCStart( nStart ); 986 return sal_True; 987 } 988 989 // Get a dir entry. 990 991 void* StgDirStrm::GetEntry( sal_Int32 n, sal_Bool bDirty ) 992 { 993 if( n < 0 ) 994 return NULL; 995 996 n *= STGENTRY_SIZE; 997 if( n < 0 && n >= nSize ) 998 return NULL; 999 return GetPtr( n, sal_True, bDirty ); 1000 } 1001 1002 // Find a dir entry. 1003 1004 StgDirEntry* StgDirStrm::Find( StgDirEntry& rStg, const String& rName ) 1005 { 1006 if( rStg.pDown ) 1007 { 1008 StgEntry aEntry; 1009 aEntry.Init(); 1010 if( !aEntry.SetName( rName ) ) 1011 { 1012 rIo.SetError( SVSTREAM_GENERALERROR ); 1013 return NULL; 1014 } 1015 // Look in the directory attached to the entry 1016 StgDirEntry aTest( aEntry ); 1017 return (StgDirEntry*) rStg.pDown->Find( &aTest ); 1018 } 1019 else 1020 return NULL; 1021 } 1022 1023 // Create a new entry. 1024 1025 StgDirEntry* StgDirStrm::Create 1026 ( StgDirEntry& rStg, const String& rName, StgEntryType eType ) 1027 { 1028 StgEntry aEntry; 1029 aEntry.Init(); 1030 aEntry.SetType( eType ); 1031 if( !aEntry.SetName( rName ) ) 1032 { 1033 rIo.SetError( SVSTREAM_GENERALERROR ); 1034 return NULL; 1035 } 1036 StgDirEntry* pRes = Find( rStg, rName ); 1037 if( pRes ) 1038 { 1039 if( !pRes->bInvalid ) 1040 { 1041 rIo.SetError( SVSTREAM_CANNOT_MAKE ); 1042 return NULL; 1043 } 1044 pRes->bInvalid = 1045 pRes->bRemoved = 1046 pRes->bTemp = sal_False; 1047 pRes->bCreated = 1048 pRes->bDirty = sal_True; 1049 } 1050 else 1051 { 1052 pRes = new StgDirEntry( aEntry ); 1053 if( StgAvlNode::Insert( (StgAvlNode**) &rStg.pDown, pRes ) ) 1054 { 1055 pRes->pUp = &rStg; 1056 pRes->ppRoot = &pRoot; 1057 pRes->bCreated = 1058 pRes->bDirty = sal_True; 1059 } 1060 else 1061 { 1062 rIo.SetError( SVSTREAM_CANNOT_MAKE ); 1063 delete pRes; pRes = NULL; 1064 } 1065 } 1066 return pRes; 1067 } 1068 1069 // Rename the given entry. 1070 1071 sal_Bool StgDirStrm::Rename( StgDirEntry& rStg, const String& rOld, const String& rNew ) 1072 { 1073 StgDirEntry* p = Find( rStg, rOld ); 1074 if( p ) 1075 { 1076 1077 if( !StgAvlNode::Remove( (StgAvlNode**) &rStg.pDown, p, sal_False ) ) 1078 return sal_False; 1079 p->aEntry.SetName( rNew ); 1080 if( !StgAvlNode::Insert( (StgAvlNode**) &rStg.pDown, p ) ) 1081 return sal_False; 1082 p->bRenamed = p->bDirty = sal_True; 1083 return sal_True; 1084 } 1085 else 1086 { 1087 rIo.SetError( SVSTREAM_FILE_NOT_FOUND ); 1088 return sal_False; 1089 } 1090 } 1091 1092 // Move the given entry to a different storage. 1093 1094 sal_Bool StgDirStrm::Move( StgDirEntry& rStg1, StgDirEntry& rStg2, const String& rName ) 1095 { 1096 StgDirEntry* p = Find( rStg1, rName ); 1097 if( p ) 1098 { 1099 if( !StgAvlNode::Move 1100 ( (StgAvlNode**) &rStg1.pDown, (StgAvlNode**) &rStg2.pDown, p ) ) 1101 return sal_False; 1102 p->bDirty = sal_True; 1103 return sal_True; 1104 } 1105 else 1106 { 1107 rIo.SetError( SVSTREAM_FILE_NOT_FOUND ); 1108 return sal_False; 1109 } 1110 } 1111 1112