/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (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.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is mozilla.org code. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 1999 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * Alternatively, the contents of this file may be used under the terms of * either of the GNU General Public License Version 2 or later (the "GPL"), * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #ifndef _MDB_ #include "mdb.h" #endif #ifndef _MORK_ #include "mork.h" #endif #ifndef _MORKNODE_ #include "morkNode.h" #endif #ifndef _MORKENV_ #include "morkEnv.h" #endif #ifndef _MORKCURSOR_ #include "morkCursor.h" #endif #ifndef _MORKPORTTABLECURSOR_ #include "morkPortTableCursor.h" #endif #ifndef _MORKSTORE_ #include "morkStore.h" #endif //3456789_123456789_123456789_123456789_123456789_123456789_123456789_123456789 // ````` ````` ````` ````` ````` // { ===== begin morkNode interface ===== /*public virtual*/ void morkPortTableCursor::CloseMorkNode(morkEnv* ev) // ClosePortTableCursor() only if open { if ( this->IsOpenNode() ) { this->MarkClosing(); this->ClosePortTableCursor(ev); this->MarkShut(); } } /*public virtual*/ morkPortTableCursor::~morkPortTableCursor() // ClosePortTableCursor() executed earlier { CloseMorkNode(mMorkEnv); } /*public non-poly*/ morkPortTableCursor::morkPortTableCursor(morkEnv* ev, const morkUsage& inUsage, nsIMdbHeap* ioHeap, morkStore* ioStore, mdb_scope inRowScope, mdb_kind inTableKind, nsIMdbHeap* ioSlotHeap) : morkCursor(ev, inUsage, ioHeap) , mPortTableCursor_Store( 0 ) , mPortTableCursor_RowScope( (mdb_scope) -1 ) // we want != inRowScope , mPortTableCursor_TableKind( (mdb_kind) -1 ) // we want != inTableKind , mPortTableCursor_LastTable ( 0 ) // not refcounted , mPortTableCursor_RowSpace( 0 ) // strong ref to row space , mPortTableCursor_TablesDidEnd( morkBool_kFalse ) , mPortTableCursor_SpacesDidEnd( morkBool_kFalse ) { if ( ev->Good() ) { if ( ioStore && ioSlotHeap ) { mCursor_Pos = -1; mCursor_Seed = 0; // let the iterator do it's own seed handling morkStore::SlotWeakStore(ioStore, ev, &mPortTableCursor_Store); if ( this->SetRowScope(ev, inRowScope) ) this->SetTableKind(ev, inTableKind); if ( ev->Good() ) mNode_Derived = morkDerived_kPortTableCursor; } else ev->NilPointerError(); } } NS_IMPL_ISUPPORTS_INHERITED1(morkPortTableCursor, morkCursor, nsIMdbPortTableCursor) morkEnv* morkPortTableCursor::CanUsePortTableCursor(nsIMdbEnv* mev, mork_bool inMutable, mdb_err* outErr) const { morkEnv* outEnv = 0; morkEnv* ev = morkEnv::FromMdbEnv(mev); if ( ev ) { if ( IsPortTableCursor() ) outEnv = ev; else NonPortTableCursorTypeError(ev); *outErr = ev->AsErr(); } MORK_ASSERT(outEnv); return outEnv; } /*public non-poly*/ void morkPortTableCursor::ClosePortTableCursor(morkEnv* ev) { if ( this ) { if ( this->IsNode() ) { mCursor_Pos = -1; mCursor_Seed = 0; mPortTableCursor_LastTable = 0; morkStore::SlotWeakStore((morkStore*) 0, ev, &mPortTableCursor_Store); morkRowSpace::SlotStrongRowSpace((morkRowSpace*) 0, ev, &mPortTableCursor_RowSpace); this->CloseCursor(ev); this->MarkShut(); } else this->NonNodeError(ev); } else ev->NilPointerError(); } // } ===== end morkNode methods ===== // ````` ````` ````` ````` ````` /*static*/ void morkPortTableCursor::NilCursorStoreError(morkEnv* ev) { ev->NewError("nil mPortTableCursor_Store"); } /*static*/ void morkPortTableCursor::NonPortTableCursorTypeError(morkEnv* ev) { ev->NewError("non morkPortTableCursor"); } mork_bool morkPortTableCursor::SetRowScope(morkEnv* ev, mork_scope inRowScope) { mPortTableCursor_RowScope = inRowScope; mPortTableCursor_LastTable = 0; // restart iteration of space mPortTableCursor_TableIter.CloseMapIter(ev); mPortTableCursor_TablesDidEnd = morkBool_kTrue; mPortTableCursor_SpacesDidEnd = morkBool_kTrue; morkStore* store = mPortTableCursor_Store; if ( store ) { morkRowSpace* space = mPortTableCursor_RowSpace; if ( inRowScope ) // intend to cover a specific scope only? { space = store->LazyGetRowSpace(ev, inRowScope); morkRowSpace::SlotStrongRowSpace(space, ev, &mPortTableCursor_RowSpace); // We want mPortTableCursor_SpacesDidEnd == morkBool_kTrue // to show this is the only space to be covered. } else // prepare space map iter to cover all space scopes { morkRowSpaceMapIter* rsi = &mPortTableCursor_SpaceIter; rsi->InitRowSpaceMapIter(ev, &store->mStore_RowSpaces); space = 0; (void) rsi->FirstRowSpace(ev, (mork_scope*) 0, &space); morkRowSpace::SlotStrongRowSpace(space, ev, &mPortTableCursor_RowSpace); if ( space ) // found first space in store mPortTableCursor_SpacesDidEnd = morkBool_kFalse; } this->init_space_tables_map(ev); } else this->NilCursorStoreError(ev); return ev->Good(); } void morkPortTableCursor::init_space_tables_map(morkEnv* ev) { morkRowSpace* space = mPortTableCursor_RowSpace; if ( space && ev->Good() ) { morkTableMapIter* ti = &mPortTableCursor_TableIter; ti->InitTableMapIter(ev, &space->mRowSpace_Tables); if ( ev->Good() ) mPortTableCursor_TablesDidEnd = morkBool_kFalse; } } mork_bool morkPortTableCursor::SetTableKind(morkEnv* ev, mork_kind inTableKind) { mPortTableCursor_TableKind = inTableKind; mPortTableCursor_LastTable = 0; // restart iteration of space mPortTableCursor_TablesDidEnd = morkBool_kTrue; morkRowSpace* space = mPortTableCursor_RowSpace; if ( !space && mPortTableCursor_RowScope == 0 ) { this->SetRowScope(ev, 0); space = mPortTableCursor_RowSpace; } this->init_space_tables_map(ev); return ev->Good(); } morkRowSpace* morkPortTableCursor::NextSpace(morkEnv* ev) { morkRowSpace* outSpace = 0; mPortTableCursor_LastTable = 0; mPortTableCursor_SpacesDidEnd = morkBool_kTrue; mPortTableCursor_TablesDidEnd = morkBool_kTrue; if ( !mPortTableCursor_RowScope ) // not just one scope? { morkStore* store = mPortTableCursor_Store; if ( store ) { morkRowSpaceMapIter* rsi = &mPortTableCursor_SpaceIter; (void) rsi->NextRowSpace(ev, (mork_scope*) 0, &outSpace); morkRowSpace::SlotStrongRowSpace(outSpace, ev, &mPortTableCursor_RowSpace); if ( outSpace ) // found next space in store { mPortTableCursor_SpacesDidEnd = morkBool_kFalse; this->init_space_tables_map(ev); if ( ev->Bad() ) outSpace = 0; } } else this->NilCursorStoreError(ev); } return outSpace; } morkTable * morkPortTableCursor::NextTable(morkEnv* ev) { mork_kind kind = mPortTableCursor_TableKind; do // until spaces end, or until we find a table in a space { morkRowSpace* space = mPortTableCursor_RowSpace; if ( mPortTableCursor_TablesDidEnd ) // current space exhausted? space = this->NextSpace(ev); // go on to the next space if ( space ) // have a space remaining that might hold tables? { #ifdef MORK_BEAD_OVER_NODE_MAPS morkTableMapIter* ti = &mPortTableCursor_TableIter; morkTable* table = ( mPortTableCursor_LastTable )? ti->NextTable(ev) : ti->FirstTable(ev); for ( ; table && ev->Good(); table = ti->NextTable(ev) ) #else /*MORK_BEAD_OVER_NODE_MAPS*/ mork_tid* key = 0; // ignore keys in table map morkTable* table = 0; // old value table in the map morkTableMapIter* ti = &mPortTableCursor_TableIter; mork_change* c = ( mPortTableCursor_LastTable )? ti->NextTable(ev, key, &table) : ti->FirstTable(ev, key, &table); for ( ; c && ev->Good(); c = ti->NextTable(ev, key, &table) ) #endif /*MORK_BEAD_OVER_NODE_MAPS*/ { if ( table && table->IsTable() ) { if ( !kind || kind == table->mTable_Kind ) { mPortTableCursor_LastTable = table; // ti->NextTable() hence return table; } } else table->NonTableTypeWarning(ev); } mPortTableCursor_TablesDidEnd = morkBool_kTrue; // space is done mPortTableCursor_LastTable = 0; // make sure next space starts fresh } } while ( ev->Good() && !mPortTableCursor_SpacesDidEnd ); return (morkTable*) 0; } // { ----- begin table iteration methods ----- // { ===== begin nsIMdbPortTableCursor methods ===== // { ----- begin attribute methods ----- NS_IMETHODIMP morkPortTableCursor::SetPort(nsIMdbEnv* mev, nsIMdbPort* ioPort) { NS_ASSERTION(PR_FALSE,"not implemented"); return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP morkPortTableCursor::GetPort(nsIMdbEnv* mev, nsIMdbPort** acqPort) { mdb_err outErr = 0; nsIMdbPort* outPort = 0; morkEnv* ev = this->CanUsePortTableCursor(mev, /*inMutable*/ morkBool_kFalse, &outErr); if ( ev ) { if ( mPortTableCursor_Store ) outPort = mPortTableCursor_Store->AcquireStoreHandle(ev); outErr = ev->AsErr(); } if ( acqPort ) *acqPort = outPort; return outErr; } NS_IMETHODIMP morkPortTableCursor::SetRowScope(nsIMdbEnv* mev, // sets pos to -1 mdb_scope inRowScope) { mdb_err outErr = 0; morkEnv* ev = this->CanUsePortTableCursor(mev, /*inMutable*/ morkBool_kFalse, &outErr); if ( ev ) { mCursor_Pos = -1; SetRowScope(ev, inRowScope); outErr = ev->AsErr(); } return outErr; } NS_IMETHODIMP morkPortTableCursor::GetRowScope(nsIMdbEnv* mev, mdb_scope* outRowScope) { mdb_err outErr = 0; mdb_scope rowScope = 0; morkEnv* ev = this->CanUsePortTableCursor(mev, /*inMutable*/ morkBool_kFalse, &outErr); if ( ev ) { rowScope = mPortTableCursor_RowScope; outErr = ev->AsErr(); } *outRowScope = rowScope; return outErr; } // setting row scope to zero iterates over all row scopes in port NS_IMETHODIMP morkPortTableCursor::SetTableKind(nsIMdbEnv* mev, // sets pos to -1 mdb_kind inTableKind) { mdb_err outErr = 0; morkEnv* ev = this->CanUsePortTableCursor(mev, /*inMutable*/ morkBool_kFalse, &outErr); if ( ev ) { mCursor_Pos = -1; SetTableKind(ev, inTableKind); outErr = ev->AsErr(); } return outErr; } NS_IMETHODIMP morkPortTableCursor::GetTableKind(nsIMdbEnv* mev, mdb_kind* outTableKind) // setting table kind to zero iterates over all table kinds in row scope { mdb_err outErr = 0; mdb_kind tableKind = 0; morkEnv* ev = this->CanUsePortTableCursor(mev, /*inMutable*/ morkBool_kFalse, &outErr); if ( ev ) { tableKind = mPortTableCursor_TableKind; outErr = ev->AsErr(); } *outTableKind = tableKind; return outErr; } // } ----- end attribute methods ----- // { ----- begin table iteration methods ----- NS_IMETHODIMP morkPortTableCursor::NextTable( // get table at next position in the db nsIMdbEnv* mev, // context nsIMdbTable** acqTable) { mdb_err outErr = 0; nsIMdbTable* outTable = 0; morkEnv* ev = CanUsePortTableCursor(mev, /*inMutable*/ morkBool_kFalse, &outErr); if ( ev ) { morkTable* table = NextTable(ev); if ( table && ev->Good() ) outTable = table->AcquireTableHandle(ev); outErr = ev->AsErr(); } if ( acqTable ) *acqTable = outTable; return outErr; } //3456789_123456789_123456789_123456789_123456789_123456789_123456789_123456789