diff options
Diffstat (limited to '')
| -rw-r--r-- | src/tests/cache.cpp | 361 |
1 files changed, 330 insertions, 31 deletions
diff --git a/src/tests/cache.cpp b/src/tests/cache.cpp index aa71c39..112d916 100644 --- a/src/tests/cache.cpp +++ b/src/tests/cache.cpp | |||
| @@ -6,6 +6,7 @@ | |||
| 6 | #include <bu/file.h> | 6 | #include <bu/file.h> |
| 7 | #include <bu/streamstack.h> | 7 | #include <bu/streamstack.h> |
| 8 | #include <bu/readwritemutex.h> | 8 | #include <bu/readwritemutex.h> |
| 9 | #include <bu/mutexlocker.h> | ||
| 9 | 10 | ||
| 10 | #include <bu/sio.h> | 11 | #include <bu/sio.h> |
| 11 | #include <bu/string.h> | 12 | #include <bu/string.h> |
| @@ -15,6 +16,7 @@ | |||
| 15 | // | 16 | // |
| 16 | 17 | ||
| 17 | template<typename keytype, typename obtype> class CacheBase; | 18 | template<typename keytype, typename obtype> class CacheBase; |
| 19 | template<typename keytype, typename obtype> class CacheEntry; | ||
| 18 | 20 | ||
| 19 | template<typename keytype, typename obtype> | 21 | template<typename keytype, typename obtype> |
| 20 | class CacheObject | 22 | class CacheObject |
| @@ -36,6 +38,27 @@ public: | |||
| 36 | virtual keytype getKey() const=0; | 38 | virtual keytype getKey() const=0; |
| 37 | virtual int getPersistenceScore() const { return 0; } | 39 | virtual int getPersistenceScore() const { return 0; } |
| 38 | 40 | ||
| 41 | void lock() | ||
| 42 | { | ||
| 43 | pEntry->lock(); | ||
| 44 | } | ||
| 45 | |||
| 46 | void unlock() | ||
| 47 | { | ||
| 48 | pEntry->unlock(); | ||
| 49 | } | ||
| 50 | |||
| 51 | int getRefCount() const | ||
| 52 | { | ||
| 53 | return pEntry->getRefCount(); | ||
| 54 | } | ||
| 55 | |||
| 56 | bool hasChanged() const | ||
| 57 | { | ||
| 58 | return bChanged; | ||
| 59 | } | ||
| 60 | |||
| 61 | protected: | ||
| 39 | void changed( bool bChanged=true ) | 62 | void changed( bool bChanged=true ) |
| 40 | { | 63 | { |
| 41 | if( this->bChanged == false && bChanged == true ) | 64 | if( this->bChanged == false && bChanged == true ) |
| @@ -45,13 +68,16 @@ public: | |||
| 45 | } | 68 | } |
| 46 | 69 | ||
| 47 | private: | 70 | private: |
| 48 | void setCache( CacheType *pCache ) | 71 | typedef CacheEntry<keytype, obtype> Entry; |
| 72 | void setCache( CacheType *pCache, Entry *pEntry ) | ||
| 49 | { | 73 | { |
| 50 | this->pCache = pCache; | 74 | this->pCache = pCache; |
| 75 | this->pEntry = pEntry; | ||
| 51 | } | 76 | } |
| 52 | 77 | ||
| 53 | private: | 78 | private: |
| 54 | CacheType *pCache; | 79 | CacheType *pCache; |
| 80 | Entry *pEntry; | ||
| 55 | bool bChanged; | 81 | bool bChanged; |
| 56 | }; | 82 | }; |
| 57 | 83 | ||
| @@ -93,6 +119,11 @@ public: | |||
| 93 | mObject.unlock(); | 119 | mObject.unlock(); |
| 94 | } | 120 | } |
| 95 | 121 | ||
| 122 | Bu::Mutex &getMutex() | ||
| 123 | { | ||
| 124 | return mObject; | ||
| 125 | } | ||
| 126 | |||
| 96 | void incRef() | 127 | void incRef() |
| 97 | { | 128 | { |
| 98 | mEntry.lock(); | 129 | mEntry.lock(); |
| @@ -263,7 +294,13 @@ public: | |||
| 263 | 294 | ||
| 264 | return *this; | 295 | return *this; |
| 265 | } | 296 | } |
| 266 | 297 | ||
| 298 | template<typename castto> | ||
| 299 | CachePtr<keytype, castto, basetype> cast() | ||
| 300 | { | ||
| 301 | return pCache->cast<obtype, castto>( *this ); | ||
| 302 | } | ||
| 303 | |||
| 267 | bool operator==( const MyType &rhs ) const | 304 | bool operator==( const MyType &rhs ) const |
| 268 | { | 305 | { |
| 269 | return pCache == rhs.pCache && | 306 | return pCache == rhs.pCache && |
| @@ -339,6 +376,7 @@ template<typename keytype, typename obtype> | |||
| 339 | class CacheBase | 376 | class CacheBase |
| 340 | { | 377 | { |
| 341 | friend class CachePtrInterface<keytype, obtype>; | 378 | friend class CachePtrInterface<keytype, obtype>; |
| 379 | friend class CacheObject<keytype, obtype>; | ||
| 342 | public: | 380 | public: |
| 343 | CacheBase() | 381 | CacheBase() |
| 344 | { | 382 | { |
| @@ -346,13 +384,14 @@ public: | |||
| 346 | 384 | ||
| 347 | virtual ~CacheBase() | 385 | virtual ~CacheBase() |
| 348 | { | 386 | { |
| 387 | Bu::ReadWriteMutex::WriteLocker wl( mCacheEntry ); | ||
| 388 | syncChanges(); | ||
| 349 | } | 389 | } |
| 350 | 390 | ||
| 351 | typedef CacheEntry<keytype, obtype> Entry; | 391 | typedef CacheEntry<keytype, obtype> Entry; |
| 352 | typedef Bu::List<keytype> KeyList; | 392 | typedef Bu::List<keytype> KeyList; |
| 353 | 393 | ||
| 354 | 394 | CachePtr<keytype, obtype> insert( obtype *pObject ) | |
| 355 | CachePtr<keytype, obtype> insert( const obtype *pObject ) | ||
| 356 | { | 395 | { |
| 357 | Entry *pEnt = addEntry( pObject ); | 396 | Entry *pEnt = addEntry( pObject ); |
| 358 | 397 | ||
| @@ -362,9 +401,9 @@ public: | |||
| 362 | } | 401 | } |
| 363 | 402 | ||
| 364 | template<typename supertype> | 403 | template<typename supertype> |
| 365 | CachePtr<keytype, supertype, obtype> insert( const supertype *pObject ) | 404 | CachePtr<keytype, supertype, obtype> insert( supertype *pObject ) |
| 366 | { | 405 | { |
| 367 | const obtype *pCast = dynamic_cast<const obtype *>( pObject ); | 406 | obtype *pCast = dynamic_cast<obtype *>( pObject ); |
| 368 | if( pCast == NULL ) | 407 | if( pCast == NULL ) |
| 369 | throw std::bad_cast(); | 408 | throw std::bad_cast(); |
| 370 | 409 | ||
| @@ -399,9 +438,45 @@ public: | |||
| 399 | return CachePtr<keytype, supertype, obtype>( this, key ); | 438 | return CachePtr<keytype, supertype, obtype>( this, key ); |
| 400 | } | 439 | } |
| 401 | 440 | ||
| 441 | template<typename supertype, typename castto> | ||
| 442 | CachePtr<keytype, castto, obtype> cast( CachePtr<keytype, supertype, obtype> &ptr ) | ||
| 443 | { | ||
| 444 | if( ptr.pEnt ) | ||
| 445 | return CachePtr<keytype, castto, obtype>( this, ptr.pEnt, ptr.kId ); | ||
| 446 | else | ||
| 447 | return CachePtr<keytype, castto, obtype>( this, ptr.kId ); | ||
| 448 | } | ||
| 449 | |||
| 450 | void erase( const keytype &key ) | ||
| 451 | { | ||
| 452 | Bu::ReadWriteMutex::WriteLocker wl( mCacheEntry ); | ||
| 453 | if( hCacheEntry.has( key ) ) | ||
| 454 | { | ||
| 455 | Entry *pEnt = hCacheEntry.get( key ); | ||
| 456 | pEnt->mEntry.lock(); | ||
| 457 | if( pEnt->iRefCount > 0 ) | ||
| 458 | { | ||
| 459 | int iCount = pEnt->iRefCount; | ||
| 460 | pEnt->mEntry.unlock(); | ||
| 461 | throw Bu::ExceptionBase( Bu::String("Cache entry %1 cannot be erased, there are %2 active references.").arg( key ).arg( iCount ).end().getStr() ); | ||
| 462 | } | ||
| 463 | delete pEnt->pObject; | ||
| 464 | delete pEnt; | ||
| 465 | hCacheEntry.erase( key ); | ||
| 466 | } | ||
| 467 | _erase( key ); | ||
| 468 | } | ||
| 469 | |||
| 402 | virtual KeyList getKeys() const=0; | 470 | virtual KeyList getKeys() const=0; |
| 471 | virtual int getSize() const=0; | ||
| 403 | 472 | ||
| 404 | virtual void sync()=0; | 473 | void sync() |
| 474 | { | ||
| 475 | Bu::ReadWriteMutex::WriteLocker wl( mCacheEntry ); | ||
| 476 | Bu::println("Syncing storage layer..."); | ||
| 477 | _sync(); | ||
| 478 | syncChanges(); | ||
| 479 | } | ||
| 405 | 480 | ||
| 406 | void _debug() | 481 | void _debug() |
| 407 | { | 482 | { |
| @@ -428,9 +503,10 @@ protected: | |||
| 428 | 503 | ||
| 429 | void objectChanged( const keytype &k ) | 504 | void objectChanged( const keytype &k ) |
| 430 | { | 505 | { |
| 506 | Bu::ReadWriteMutex::WriteLocker wl( mCacheEntry ); | ||
| 507 | hChanged.insert( k, true ); | ||
| 431 | } | 508 | } |
| 432 | 509 | ||
| 433 | |||
| 434 | protected: | 510 | protected: |
| 435 | virtual void _create( const obtype *o )=0; | 511 | virtual void _create( const obtype *o )=0; |
| 436 | virtual void _erase( const keytype &k )=0; | 512 | virtual void _erase( const keytype &k )=0; |
| @@ -438,14 +514,17 @@ protected: | |||
| 438 | virtual obtype *_load( const keytype &k )=0; | 514 | virtual obtype *_load( const keytype &k )=0; |
| 439 | virtual void _save( const obtype *o )=0; | 515 | virtual void _save( const obtype *o )=0; |
| 440 | 516 | ||
| 517 | virtual void _sync()=0; | ||
| 518 | |||
| 441 | private: | 519 | private: |
| 442 | Entry *addEntry( const obtype *pObject ) | 520 | Entry *addEntry( obtype *pObject ) |
| 443 | { | 521 | { |
| 444 | Entry *pEnt = new Entry( const_cast<obtype *>(pObject) ); | 522 | Entry *pEnt = new Entry( pObject ); |
| 445 | mCacheEntry.lockWrite(); | 523 | mCacheEntry.lockWrite(); |
| 446 | hCacheEntry.insert( pObject->getKey(), pEnt ); | 524 | hCacheEntry.insert( pObject->getKey(), pEnt ); |
| 447 | mCacheEntry.unlockWrite(); | 525 | mCacheEntry.unlockWrite(); |
| 448 | _create( pObject ); | 526 | _create( pObject ); |
| 527 | pObject->setCache( this, pEnt ); | ||
| 449 | 528 | ||
| 450 | return pEnt; | 529 | return pEnt; |
| 451 | } | 530 | } |
| @@ -463,25 +542,60 @@ private: | |||
| 463 | // try to load the object from the backing store | 542 | // try to load the object from the backing store |
| 464 | obtype *pObject = _load( k ); | 543 | obtype *pObject = _load( k ); |
| 465 | pEnt = new Entry( pObject ); | 544 | pEnt = new Entry( pObject ); |
| 545 | pObject->setCache( this, pEnt ); | ||
| 466 | Bu::ReadWriteMutex::WriteLocker wl( mCacheEntry ); | 546 | Bu::ReadWriteMutex::WriteLocker wl( mCacheEntry ); |
| 467 | hCacheEntry.insert( k, pEnt ); | 547 | hCacheEntry.insert( k, pEnt ); |
| 468 | } | 548 | } |
| 469 | return pEnt; | 549 | return pEnt; |
| 470 | } | 550 | } |
| 471 | 551 | ||
| 552 | void syncChanges() | ||
| 553 | { | ||
| 554 | Bu::println("Syncing: %1 entries changed.").arg( hChanged.getSize() ); | ||
| 555 | if( !hChanged.isEmpty() ) | ||
| 556 | { | ||
| 557 | for( typename CacheKeySet::iterator i = hChanged.begin(); i; i++ ) | ||
| 558 | { | ||
| 559 | Entry *pEnt = hCacheEntry.get( i.getKey() ); | ||
| 560 | Bu::MutexLocker ml( pEnt->getMutex() ); | ||
| 561 | _save( pEnt->getPtr() ); | ||
| 562 | } | ||
| 563 | hChanged.clear(); | ||
| 564 | } | ||
| 565 | } | ||
| 566 | |||
| 472 | private: | 567 | private: |
| 473 | typedef Bu::Hash<keytype, Entry *> CacheEntryHash; | 568 | typedef Bu::Hash<keytype, Entry *> CacheEntryHash; |
| 569 | typedef Bu::Hash<keytype, bool> CacheKeySet; | ||
| 474 | CacheEntryHash hCacheEntry; | 570 | CacheEntryHash hCacheEntry; |
| 571 | CacheKeySet hChanged; | ||
| 475 | Bu::ReadWriteMutex mCacheEntry; | 572 | Bu::ReadWriteMutex mCacheEntry; |
| 476 | }; | 573 | }; |
| 477 | 574 | ||
| 575 | template<typename obtype> | ||
| 576 | void _cacheObjectSave( Bu::Stream &s, obtype *pObject ) | ||
| 577 | { | ||
| 578 | Bu::Archive ar( s, Bu::Archive::save ); | ||
| 579 | ar << *pObject; | ||
| 580 | } | ||
| 581 | |||
| 582 | template<typename obtype> | ||
| 583 | obtype *_cacheObjectLoad( Bu::Stream &s ) | ||
| 584 | { | ||
| 585 | Bu::Archive ar( s, Bu::Archive::load ); | ||
| 586 | obtype *ret = new obtype(); | ||
| 587 | ar >> *ret; | ||
| 588 | return ret; | ||
| 589 | } | ||
| 590 | |||
| 478 | template<typename keytype, typename obtype> | 591 | template<typename keytype, typename obtype> |
| 479 | class MyriadCache : public CacheBase<keytype, obtype> | 592 | class MyriadCache : public CacheBase<keytype, obtype> |
| 480 | { | 593 | { |
| 481 | public: | 594 | public: |
| 482 | MyriadCache( Bu::Stream &sStore, int iBlockSize=512, int iPreallocate=8 ) : | 595 | MyriadCache( Bu::Stream &sStore, int iBlockSize=512, int iPreallocate=8 ) : |
| 483 | sStore( sStore ), | 596 | sStore( sStore ), |
| 484 | mStore( sStore, iBlockSize, iPreallocate ) | 597 | mStore( sStore, iBlockSize, iPreallocate ), |
| 598 | bStructureChanged( false ) | ||
| 485 | { | 599 | { |
| 486 | try | 600 | try |
| 487 | { | 601 | { |
| @@ -502,13 +616,14 @@ public: | |||
| 502 | if( mStore.createStreamWithId( 1 ) != 1 ) | 616 | if( mStore.createStreamWithId( 1 ) != 1 ) |
| 503 | throw Bu::ExceptionBase("Error creating index stream."); | 617 | throw Bu::ExceptionBase("Error creating index stream."); |
| 504 | 618 | ||
| 505 | sync(); | 619 | _sync(); |
| 506 | } | 620 | } |
| 507 | } | 621 | } |
| 508 | 622 | ||
| 509 | virtual ~MyriadCache() | 623 | virtual ~MyriadCache() |
| 510 | { | 624 | { |
| 511 | sync(); | 625 | Bu::println("MyriadCache destroying."); |
| 626 | _sync(); | ||
| 512 | } | 627 | } |
| 513 | 628 | ||
| 514 | using typename CacheBase<keytype,obtype>::KeyList; | 629 | using typename CacheBase<keytype,obtype>::KeyList; |
| @@ -518,47 +633,60 @@ public: | |||
| 518 | Bu::ReadWriteMutex::ReadLocker rl( rwStore ); | 633 | Bu::ReadWriteMutex::ReadLocker rl( rwStore ); |
| 519 | return hIndex.getKeys(); | 634 | return hIndex.getKeys(); |
| 520 | } | 635 | } |
| 521 | 636 | ||
| 522 | virtual void sync() | 637 | virtual int getSize() const |
| 523 | { | 638 | { |
| 524 | Bu::ReadWriteMutex::ReadLocker wl( rwStore ); | 639 | Bu::ReadWriteMutex::ReadLocker rl( rwStore ); |
| 525 | Bu::MyriadStream ms = mStore.openStream( 1 ); | 640 | return hIndex.getSize(); |
| 526 | Bu::Archive ar( ms, Bu::Archive::save ); | ||
| 527 | ar << (uint8_t)0 << hIndex; | ||
| 528 | ar.close(); | ||
| 529 | ms.setSize( ms.tell() ); | ||
| 530 | } | 641 | } |
| 531 | 642 | ||
| 532 | protected: | 643 | protected: |
| 533 | virtual void _create( const obtype *o ) | 644 | virtual void _create( const obtype *o ) |
| 534 | { | 645 | { |
| 646 | Bu::ReadWriteMutex::WriteLocker wl( rwStore ); | ||
| 535 | hIndex.insert( o->getKey(), mStore.createStream() ); | 647 | hIndex.insert( o->getKey(), mStore.createStream() ); |
| 536 | _save( o ); | 648 | _save( o ); |
| 649 | |||
| 650 | bStructureChanged = true; | ||
| 537 | } | 651 | } |
| 538 | 652 | ||
| 539 | virtual void _erase( const keytype &k ) | 653 | virtual void _erase( const keytype &k ) |
| 540 | { | 654 | { |
| 655 | Bu::ReadWriteMutex::WriteLocker wl( rwStore ); | ||
| 541 | mStore.deleteStream( hIndex.get( k ) ); | 656 | mStore.deleteStream( hIndex.get( k ) ); |
| 542 | hIndex.erase( k ); | 657 | hIndex.erase( k ); |
| 658 | |||
| 659 | bStructureChanged = true; | ||
| 543 | } | 660 | } |
| 544 | 661 | ||
| 545 | virtual obtype *_load( const keytype &k ) | 662 | virtual obtype *_load( const keytype &k ) |
| 546 | { | 663 | { |
| 547 | Bu::MyriadStream ms = mStore.openStream( hIndex.get( k ) ); | 664 | Bu::MyriadStream ms = mStore.openStream( hIndex.get( k ) ); |
| 548 | Bu::Archive ar( ms, Bu::Archive::load ); | 665 | return _cacheObjectLoad<obtype>( ms ); |
| 549 | obtype *ret = new obtype(); | ||
| 550 | ar >> *ret; | ||
| 551 | return ret; | ||
| 552 | } | 666 | } |
| 553 | 667 | ||
| 554 | virtual void _save( const obtype *o ) | 668 | virtual void _save( const obtype *o ) |
| 555 | { | 669 | { |
| 556 | Bu::MyriadStream ms = mStore.openStream( hIndex.get( o->getKey() ) ); | 670 | Bu::MyriadStream ms = mStore.openStream( hIndex.get( o->getKey() ) ); |
| 557 | { | 671 | _cacheObjectSave( ms, o ); |
| 558 | Bu::Archive ar( ms, Bu::Archive::save ); | 672 | ms.setSize( ms.tell() ); |
| 559 | ar << *o; | 673 | } |
| 560 | } | 674 | |
| 675 | virtual void _sync() | ||
| 676 | { | ||
| 677 | Bu::println("Syncing myriad header."); | ||
| 678 | Bu::ReadWriteMutex::ReadLocker wl( rwStore ); | ||
| 679 | if( !bStructureChanged ) | ||
| 680 | return; | ||
| 681 | |||
| 682 | Bu::println(" --> Header changed, write required."); | ||
| 683 | Bu::MyriadStream ms = mStore.openStream( 1 ); | ||
| 684 | Bu::Archive ar( ms, Bu::Archive::save ); | ||
| 685 | ar << (uint8_t)0 << hIndex; | ||
| 686 | ar.close(); | ||
| 561 | ms.setSize( ms.tell() ); | 687 | ms.setSize( ms.tell() ); |
| 688 | |||
| 689 | bStructureChanged = false; | ||
| 562 | } | 690 | } |
| 563 | 691 | ||
| 564 | private: | 692 | private: |
| @@ -566,6 +694,7 @@ private: | |||
| 566 | Bu::Myriad mStore; | 694 | Bu::Myriad mStore; |
| 567 | Bu::Hash<keytype, int> hIndex; | 695 | Bu::Hash<keytype, int> hIndex; |
| 568 | mutable Bu::ReadWriteMutex rwStore; | 696 | mutable Bu::ReadWriteMutex rwStore; |
| 697 | bool bStructureChanged; | ||
| 569 | }; | 698 | }; |
| 570 | 699 | ||
| 571 | // | 700 | // |
| @@ -596,16 +725,72 @@ public: | |||
| 596 | return uId; | 725 | return uId; |
| 597 | } | 726 | } |
| 598 | 727 | ||
| 599 | Bu::String getName() | 728 | Bu::String getName() const |
| 600 | { | 729 | { |
| 601 | return sName; | 730 | return sName; |
| 602 | } | 731 | } |
| 603 | 732 | ||
| 733 | void setName( const Bu::String &sNewName ) | ||
| 734 | { | ||
| 735 | sName = sNewName; | ||
| 736 | changed(); | ||
| 737 | } | ||
| 738 | |||
| 739 | virtual Bu::String toString() const=0; | ||
| 740 | |||
| 604 | private: | 741 | private: |
| 605 | Bu::Uuid uId; | 742 | Bu::Uuid uId; |
| 606 | Bu::String sName; | 743 | Bu::String sName; |
| 607 | }; | 744 | }; |
| 608 | 745 | ||
| 746 | class SubSomethingA : public Something | ||
| 747 | { | ||
| 748 | friend Bu::ArchiveBase &operator>>( Bu::ArchiveBase &ar, SubSomethingA &s ); | ||
| 749 | friend Bu::ArchiveBase &operator<<( Bu::ArchiveBase &ar, const SubSomethingA &s ); | ||
| 750 | public: | ||
| 751 | SubSomethingA() | ||
| 752 | { | ||
| 753 | } | ||
| 754 | |||
| 755 | SubSomethingA( const Bu::String &sName, int iNumber ) : | ||
| 756 | Something( sName ), | ||
| 757 | iNumber( iNumber ) | ||
| 758 | { | ||
| 759 | } | ||
| 760 | |||
| 761 | virtual Bu::String toString() const | ||
| 762 | { | ||
| 763 | return Bu::String("[typeA] %1 (%2)").arg( getName() ).arg( iNumber ); | ||
| 764 | } | ||
| 765 | |||
| 766 | private: | ||
| 767 | int iNumber; | ||
| 768 | }; | ||
| 769 | |||
| 770 | class SubSomethingB : public Something | ||
| 771 | { | ||
| 772 | friend Bu::ArchiveBase &operator>>( Bu::ArchiveBase &ar, SubSomethingB &s ); | ||
| 773 | friend Bu::ArchiveBase &operator<<( Bu::ArchiveBase &ar, const SubSomethingB &s ); | ||
| 774 | public: | ||
| 775 | SubSomethingB() | ||
| 776 | { | ||
| 777 | } | ||
| 778 | |||
| 779 | SubSomethingB( const Bu::String &sName, const Bu::String &sString ) : | ||
| 780 | Something( sName ), | ||
| 781 | sString( sString ) | ||
| 782 | { | ||
| 783 | } | ||
| 784 | |||
| 785 | virtual Bu::String toString() const | ||
| 786 | { | ||
| 787 | return Bu::String("[typeB] %1 (%2)").arg( getName() ).arg( sString ); | ||
| 788 | } | ||
| 789 | |||
| 790 | private: | ||
| 791 | Bu::String sString; | ||
| 792 | }; | ||
| 793 | |||
| 609 | Bu::ArchiveBase &operator>>( Bu::ArchiveBase &ar, Something &s ) | 794 | Bu::ArchiveBase &operator>>( Bu::ArchiveBase &ar, Something &s ) |
| 610 | { | 795 | { |
| 611 | return ar >> s.uId >> s.sName; | 796 | return ar >> s.uId >> s.sName; |
| @@ -616,7 +801,77 @@ Bu::ArchiveBase &operator<<( Bu::ArchiveBase &ar, const Something &s ) | |||
| 616 | return ar << s.uId << s.sName; | 801 | return ar << s.uId << s.sName; |
| 617 | } | 802 | } |
| 618 | 803 | ||
| 804 | Bu::ArchiveBase &operator>>( Bu::ArchiveBase &ar, SubSomethingA &s ) | ||
| 805 | { | ||
| 806 | return ar >> (Something &)s >> s.iNumber; | ||
| 807 | } | ||
| 808 | |||
| 809 | Bu::ArchiveBase &operator<<( Bu::ArchiveBase &ar, const SubSomethingA &s ) | ||
| 810 | { | ||
| 811 | return ar << (Something &)s << s.iNumber; | ||
| 812 | } | ||
| 813 | |||
| 814 | Bu::ArchiveBase &operator>>( Bu::ArchiveBase &ar, SubSomethingB &s ) | ||
| 815 | { | ||
| 816 | return ar >> (Something &)s >> s.sString; | ||
| 817 | } | ||
| 818 | |||
| 819 | Bu::ArchiveBase &operator<<( Bu::ArchiveBase &ar, const SubSomethingB &s ) | ||
| 820 | { | ||
| 821 | return ar << (Something &)s << s.sString; | ||
| 822 | } | ||
| 823 | |||
| 824 | template<> | ||
| 825 | void _cacheObjectSave<const Something>( Bu::Stream &s, const Something *pObject ) | ||
| 826 | { | ||
| 827 | Bu::Archive ar( s, Bu::Archive::save ); | ||
| 828 | if( typeid(*pObject) == typeid(SubSomethingA) ) | ||
| 829 | { | ||
| 830 | ar << (uint8_t)1 << (SubSomethingA &)*pObject; | ||
| 831 | } | ||
| 832 | else if( typeid(*pObject) == typeid(SubSomethingB) ) | ||
| 833 | { | ||
| 834 | ar << (uint8_t)2 << (SubSomethingB &)*pObject; | ||
| 835 | } | ||
| 836 | else | ||
| 837 | { | ||
| 838 | Bu::println("Not a recognized type!"); | ||
| 839 | throw Bu::ExceptionBase("Not recognized type!"); | ||
| 840 | } | ||
| 841 | } | ||
| 842 | |||
| 843 | template<> | ||
| 844 | Something *_cacheObjectLoad<Something>( Bu::Stream &s ) | ||
| 845 | { | ||
| 846 | Bu::Archive ar( s, Bu::Archive::load ); | ||
| 847 | uint8_t uType; | ||
| 848 | ar >> uType; | ||
| 849 | switch( uType ) | ||
| 850 | { | ||
| 851 | case 1: | ||
| 852 | { | ||
| 853 | SubSomethingA *ret = new SubSomethingA(); | ||
| 854 | ar >> *ret; | ||
| 855 | return ret; | ||
| 856 | } | ||
| 857 | break; | ||
| 858 | |||
| 859 | case 2: | ||
| 860 | { | ||
| 861 | SubSomethingB *ret = new SubSomethingB(); | ||
| 862 | ar >> *ret; | ||
| 863 | return ret; | ||
| 864 | } | ||
| 865 | break; | ||
| 866 | |||
| 867 | default: | ||
| 868 | throw Bu::ExceptionBase("Flagrant error! Invalid type!"); | ||
| 869 | } | ||
| 870 | } | ||
| 871 | |||
| 619 | typedef CachePtr<Bu::Uuid, Something> SomethingPtr; | 872 | typedef CachePtr<Bu::Uuid, Something> SomethingPtr; |
| 873 | typedef CachePtr<Bu::Uuid, SubSomethingA, Something> SomethingAPtr; | ||
| 874 | typedef CachePtr<Bu::Uuid, SubSomethingB, Something> SomethingBPtr; | ||
| 620 | typedef MyriadCache<Bu::Uuid, Something> SomethingCache; | 875 | typedef MyriadCache<Bu::Uuid, Something> SomethingCache; |
| 621 | 876 | ||
| 622 | int main( int argc, char *argv[] ) | 877 | int main( int argc, char *argv[] ) |
| @@ -624,6 +879,39 @@ int main( int argc, char *argv[] ) | |||
| 624 | Bu::File fStore("test.myr", Bu::File::Create|Bu::File::ReadWrite); | 879 | Bu::File fStore("test.myr", Bu::File::Create|Bu::File::ReadWrite); |
| 625 | SomethingCache c( fStore ); | 880 | SomethingCache c( fStore ); |
| 626 | 881 | ||
| 882 | SomethingPtr ptr; | ||
| 883 | if( time(NULL)%2 ) | ||
| 884 | ptr = c.insert( new SubSomethingA("Hello", 55) ).cast<Something>(); | ||
| 885 | else | ||
| 886 | ptr = c.insert( new SubSomethingB("Goodbye", "Things") ).cast<Something>(); | ||
| 887 | |||
| 888 | Bu::println("Something[%1]: %2"). | ||
| 889 | arg( ptr.getKey() ). | ||
| 890 | arg( ptr->getName() ); | ||
| 891 | |||
| 892 | SomethingCache::KeyList lKeys = c.getKeys(); | ||
| 893 | Bu::println("Count: %1").arg( lKeys.getSize() ); | ||
| 894 | int j = 0; | ||
| 895 | for( SomethingCache::KeyList::iterator i = lKeys.begin(); i; i++ ) | ||
| 896 | { | ||
| 897 | Bu::println(" - %1: '%2'").arg( *i ).arg( c.get( *i )->toString() ); | ||
| 898 | } | ||
| 899 | Bu::println("Count: %1").arg( c.getSize() ); | ||
| 900 | |||
| 901 | SomethingPtr p2 = c.get( ptr.getKey() ); | ||
| 902 | Bu::println("%1 == %2").arg( p2->getName() ).arg( ptr->getName() ); | ||
| 903 | SomethingPtr p3 = ptr.cast<Something>(); | ||
| 904 | Bu::println("%1: %2").arg( p3.getKey() ).arg( p3->getName() ); | ||
| 905 | |||
| 906 | return 0; | ||
| 907 | } | ||
| 908 | |||
| 909 | /* | ||
| 910 | int main( int argc, char *argv[] ) | ||
| 911 | { | ||
| 912 | Bu::File fStore("test.myr", Bu::File::Create|Bu::File::ReadWrite); | ||
| 913 | SomethingCache c( fStore ); | ||
| 914 | |||
| 627 | for( int j = 1; j < argc; j++ ) | 915 | for( int j = 1; j < argc; j++ ) |
| 628 | { | 916 | { |
| 629 | SomethingPtr ptr = c.insert( new Something(argv[j]) ); | 917 | SomethingPtr ptr = c.insert( new Something(argv[j]) ); |
| @@ -634,10 +922,15 @@ int main( int argc, char *argv[] ) | |||
| 634 | } | 922 | } |
| 635 | 923 | ||
| 636 | SomethingCache::KeyList lKeys = c.getKeys(); | 924 | SomethingCache::KeyList lKeys = c.getKeys(); |
| 925 | Bu::println("Count: %1").arg( lKeys.getSize() ); | ||
| 926 | int j = 0; | ||
| 637 | for( SomethingCache::KeyList::iterator i = lKeys.begin(); i; i++ ) | 927 | for( SomethingCache::KeyList::iterator i = lKeys.begin(); i; i++ ) |
| 638 | { | 928 | { |
| 639 | Bu::println(" - %1: '%2'").arg( *i ).arg( c.get( *i )->getName() ); | 929 | Bu::println(" - %1: '%2'").arg( *i ).arg( c.get( *i )->getName() ); |
| 930 | if( ((j++)%2) ) | ||
| 931 | c.erase( *i ); | ||
| 640 | } | 932 | } |
| 933 | Bu::println("Count: %1").arg( c.getSize() ); | ||
| 641 | 934 | ||
| 642 | c._debug(); | 935 | c._debug(); |
| 643 | 936 | ||
| @@ -673,12 +966,18 @@ int main( int argc, char *argv[] ) | |||
| 673 | 966 | ||
| 674 | c._debug(); | 967 | c._debug(); |
| 675 | 968 | ||
| 969 | { | ||
| 676 | SomethingPtr::Locker l( p3 ); | 970 | SomethingPtr::Locker l( p3 ); |
| 677 | 971 | ||
| 678 | Bu::println("Name again: %1").arg( p3->getName() ); | 972 | Bu::println("Name again: %1").arg( p3->getName() ); |
| 679 | 973 | ||
| 974 | p3->setName( p3->getName() + " - again" ); | ||
| 975 | } | ||
| 976 | |||
| 977 | c.sync(); | ||
| 978 | |||
| 680 | c._debug(); | 979 | c._debug(); |
| 681 | 980 | ||
| 682 | return 0; | 981 | return 0; |
| 683 | } | 982 | } |
| 684 | 983 | */ | |
