Logo Search packages:      
Sourcecode: maxdb-7.5.00 version File versions

vbd500.cpp

/*!**************************************************************************

  module      : vbd500.cpp
  special area: Tree
  responsible : UweH
  last changed: 2000-08-11  8:38
  see also    : example.html ...
  copyright:    (c) 1998-2004 SAP AG
  description : methods for navigating through the tree structure



    ========== licence begin  GPL
    Copyright (c) 1998-2005 SAP AG

    This program is free software; you can redistribute it and/or
    modify it under the terms of the GNU General Public License
    as published by the Free Software Foundation; either version 2
    of the License, or (at your option) any later version.

    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with this program; if not, write to the Free Software
    Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
    ========== licence end


*****************************************************************************/


/*===========================================================================*
 *  INCLUDES                                                                 *
 *===========================================================================*/

#include <string.h>

#include "gsp00.h"
#include "ggg00.h"
#include "gbd00.h"
#include "gbd500.h"
#include "gbd501.h"
#include "gsp03.h"
#include "gsp03_3.h"


#include "hbd20_2.h" // bd20IsPageRequested
#include "heo51.h"   // vgetpid
#include "heo56.h"   // vsleep
#include "hgg01_1.h" // g01abort,g01opmsg
#include "hgg01_3.h" // g01vtrace
#include "hgg05.h"   // gg05KeyAssign   PTS 1103975 JA 1999-09-15 
#include "hsp77.h"   // sp77sprintf

#include "Trace/Trace_Entry.hpp"

#if COMPILEMODE_MEO00 >= SLOW_MEO00 
#include "hta99.h"
#endif

/*===========================================================================*
 *  DEFINES                                                                  *
 *===========================================================================*/
#define NIL_REC_INDEX -1
/*===========================================================================*
 *  MACROS                                                                   *
 *===========================================================================*/

/*===========================================================================*
 *  LOCAL CLASSES, STRUCTURES, TYPES, UNIONS ...                             *
 *===========================================================================*/

/*===========================================================================*
 *  EXTERNAL VARIABLES                                                       *
 *===========================================================================*/

/*===========================================================================*
 *  GLOBAL VARIABLES                                                         *
 *===========================================================================*/

/*===========================================================================*
 *  LOCAL VARIABLES                                                          *
 *===========================================================================*/

/*===========================================================================*
 *  LOCAL FUNCTIONS (PROTOTYPES)                                             *
 *===========================================================================*/

/*===========================================================================*
 *  DEFINITION OF METHODS DECLARED IN gbd500.h (CODE)                        * 
 *===========================================================================*/

/*---------------------------------------------------------------------------*/

inline bool
cbd500_Tree::bd500_BeyondStopKey (bool FromNextNode) const
{
    ROUTINE_DBG_MEO00 ("bd500_BeyondStopKey");

    if (NULL == m_pStopKey) return false;

    if ( FromNextNode )
    {
        tgg00_RecPtr      pRec = m_Node.bd600RecPtr(FIRST_REC_INDEX_BD00);
        tsp00_LcompResult CompResult;
        
        s30cmp (pRec->recKey_gg00().keyVal_gg00(), POS_OFF_DIFF_BD00, pRec->recKeyLen_gg00(),
            m_pStopKey, POS_OFF_DIFF_BD00, m_StopKeyLen, 
            CompResult);

        return l_greater == CompResult;
    }
    else
    {
        tgg00_RecPtr      pRec = m_Node.bd600RecPtr (m_Node.bd600MaxRecIndex());
        tsp00_LcompResult CompResult;
        
        s30cmp (pRec->recKey_gg00().keyVal_gg00(), POS_OFF_DIFF_BD00, pRec->recKeyLen_gg00(),
            m_pStopKey, POS_OFF_DIFF_BD00, m_StopKeyLen, 
            CompResult);

        return l_less == CompResult;
    }
}

/*---------------------------------------------------------------------------*/

void
00132 cbd500_Tree::bd500FindBranch (cbd600_Node  &Node,
                              tsp00_KeyPtr  pSepKey,
                              tsp00_Int4    SepKeyLen,
                              tsp00_Int4   &SepRecIndex)
{
    ROUTINE_DBG_MEO00 ("bd500FindBranch");

    tbd_searchresult SearchResult;

    SearchResult.becomes (nonefound);

    Node.bd600SearchRecord (pSepKey, SepKeyLen, SepRecIndex, SearchResult);
    if ( e_ok != Node.m_TrError ) return;

    switch (SearchResult)
    {
    case thisfound: break;         // separator found
    case nextfound: --SepRecIndex; // separator found
                    break;
    case lastfound: break;         // separator found
    case nonefound: break;         // impossible
    default:        break;         // impossible
    }

    if (SepRecIndex < FIRST_REC_INDEX_BD00)
    {
        Node.m_TrError = e_invalid_index_structure;
        return;
    }
                        
    if (Node.bd600RecPtr(SepRecIndex)->recLen_gg00() > sizeof (tbd00_Separator))
    {
        Node.m_TrError = e_illegal_branchlength;
    }
        return;
}


/*---------------------------------------------------------------------------*/

void cbd500_Tree::bd500_NextSubNode (tsp00_Int4       RecIndex,
                                     tsp00_Int4       TargetLevel,
                                     tbd_node_request NodeRequest)
{
    ROUTINE_DBG_MEO00 ("bd500_NextSubNode");

    if ( LEAF_LEVEL_BD00 != m_CurrLevel )
    {
        /* invalidate the next value, so NextNode is not functioning wrong */
        m_LeftMostIdNextLowerLevel = NIL_PAGE_NO_GG00;

        if (g01glob.treecheck)
            m_Node.bd600CheckIndex();

        tbd500_PathInfoElem & PathInfo   = m_PathInfo [m_CurrLevel];

        PathInfo.piId_bd500              = m_Node.m_pNode->nd_id();
        PathInfo.piSubNodeRecIndex_bd500 = RecIndex;
        PathInfo.piMaxRecIndex_bd500     = m_Node.bd600MaxRecIndex();
        PathInfo.piLeafCount_bd500       = m_Node.bd600LeafCount(RecIndex, RecIndex);

        --m_CurrLevel;

        tbd500_PathInfoElem & SubNodePathInfo = m_PathInfo [m_CurrLevel];

        SubNodePathInfo.piId_bd500 = m_Node.bd600RecPtr(RecIndex)->recPno_gg00().gg90GetInt4();

        if ( FIRST_REC_INDEX_BD00 == RecIndex )
            SubNodePathInfo.piLeftId_bd500 = NIL_PAGE_NO_GG00;
        else
            SubNodePathInfo.piLeftId_bd500 = m_Node.bd600RecPtr (RecIndex-1)->recPno_gg00().gg90GetInt4();


        m_Node.bd600Release ( nr_for_update == m_Node.m_RequestKind );

        if ( m_CurrLevel > TargetLevel )
            m_Node.bd600GetNode (SubNodePathInfo.piId_bd500, tbd_node_request::fromConst(nr_for_read));
        else
            m_Node.bd600GetNode (SubNodePathInfo.piId_bd500, NodeRequest);
    }

#   if COMPILEMODE_MEO00 >= SLOW_MEO00 
    t01int4 (bd_index, "SubNodeId:  ", (m_Node.m_pNode!=NULL)?m_Node.m_pNode->nd_id():NIL_PAGE_NO_GG00);
#   endif
}

/*---------------------------------------------------------------------------*/

void
00221 cbd500_Tree::bd500FindNode (tsp00_KeyPtr pKey,
                            tsp00_Int4   KeyLen,
                            tsp00_Int4   Level = LEAF_LEVEL_BD00)
{
    /* Preconditions:
       - The Tree is locked correctly.
       - The internal Node (m_Node) contains the NodeRequest
         for the Node/Leaf, which is searched.
       Description:
       - The Search begins always with the root
    */
    ROUTINE_DBG_MEO00 ("bd500FindNode");

    tsp00_Int4       RecIndex;
    tbd_node_request TargetRequest = m_Node.m_RequestKind;

    bd500SetToRoot(); 
    if ( e_ok != m_TrError ) return;
    bd500_Init();

    while ( m_CurrLevel > Level)
    {
        bd500_FindBranch (pKey, KeyLen, RecIndex);
        if ( e_ok != m_TrError ) break;

        bd500_NextSubNode (RecIndex, Level, TargetRequest);
        if ( e_ok != m_TrError ) break;
    }
}

/*---------------------------------------------------------------------------*/

void
00254 cbd500_Tree::bd500Continue ( const tsp00_KeyPtr rePositionKey,
                             const tsp00_Int2   rePositionKeyLen)
{
    ROUTINE_DBG_MEO00 ("bd500Continue");
    m_Node.m_Current.bd300LockTree();
    if ( e_ok == m_TrError )
        bd500FindNode (rePositionKey, rePositionKeyLen, LEAF_LEVEL_BD00);
}

/*---------------------------------------------------------------------------*/

void
cbd500_Tree::bd500_CheckStopKeyAndInterrupt (bool bAscending)
{
    if ( m_Node.bd600IsLeaf() )
    {
        if ( bd500_BeyondStopKey(bAscending) )
        {
#           if COMPILEMODE_MEO00 >= SLOW_MEO00 
            t01int4 (bd_index, "BeyondStp ID", m_Node.m_pNode->nd_id());
#           endif
            m_Node.bd600Release (IS_CHANGED_BD600);
        }
        else
        {
            if ( m_Interruptible &&  // PTS 1121770 TS 2003-04-25
                m_Node.m_Current.bd300RootPtrs().np_ptr() != 0 ) // PTS 1121589 UH 2003-04-11 (Drop has not)
            {
                if ( (nr_for_read == m_Node.m_Current.bd300RootRequestKind())
                    &&
                    bd20IsPageRequested (m_Node.m_Current.bd300RootPtrs().np_cbptr()) )
                {
                    // PTS 1121589 UH 2003-04-11 local rePositionKey added
                    tsp00_Key    rePositionKeyValue;
                    tsp00_KeyPtr rePositionKey = reinterpret_cast<tsp00_KeyPtr>(&rePositionKeyValue);
                    tsp00_Int2   rePositionKeyLen; 
                    
                    const tsp00_Int4 recIndex = bAscending ? FIRST_REC_INDEX_BD00 : m_Node.bd600MaxRecIndex();
                    bd500Break( recIndex, rePositionKey, rePositionKeyLen );

                    // TorstenS 2002-07-17: reduce data cache priority problems by rescheduling the task
                    // tsp00_TaskId taskId;  PTS 1121589 UH 2003-04-11 exchanged with trTaskId
                    // vgetpid( taskId );    PTS 1121589 UH 2003-04-11 exchanged with trTaskId
                    vsleep( m_Node.m_Current.curr_trans->trTaskId_gg00, 0 ); 
                    
                    bd500Continue ( rePositionKey, rePositionKeyLen);
                }
            }
        }
    }
}


/*---------------------------------------------------------------------------*/

void
00310 cbd500_Tree::bd500NextNode ()
{
    ROUTINE_DBG_MEO00 ("bd500NextNode");

    if ( // PTS 1103703 AK 30-08-1999 
        (m_Node.m_Current.curr_trans->trRteCommPtr_gg00->to_cancel) &&    
        (
        (m_select == m_Node.m_Current.curr_action) ||
        (m_verify == m_Node.m_Current.curr_action)
        )
       )                     
    {
        m_TrError = e_cancelled;
        g01opmsg (sp3p_knldiag, sp3m_info, csp3_bd_msg, csp3_n_btree,
              "BD500: Next cancelled   " , m_Node.m_Current.curr_trans->trTaskId_gg00);
        return;
    }

    if ( ! m_Node.bd600IsAccessable() ) return;

    if ( NIL_PAGE_NO_GG00 != m_Node.m_pNode->nd_right() )
    {
        m_PathInfo[m_CurrLevel].piLeftId_bd500 = m_Node.m_pNode->nd_id();
        m_Node.bd600Move ( m_Node.m_pNode->nd_right() );
        if ( e_ok != m_TrError ) return;
        m_PathInfo[m_CurrLevel].piSubNodeRecIndex_bd500 = FIRST_REC_INDEX_BD00;
        m_PathInfo[m_CurrLevel].piMaxRecIndex_bd500     = m_Node.bd600MaxRecIndex();
        m_PathInfo[m_CurrLevel].piLeafCount_bd500       = m_Node.bd600LeafCount(FIRST_REC_INDEX_BD00,FIRST_REC_INDEX_BD00);
    }
    else
    {
        if ( m_Node.bd600IsLeaf() )
        {
            m_Node.bd600Release (IS_CHANGED_BD600);
            return;
        }
        else
            m_Node.bd600Release (IS_CHANGED_BD600);

        if ( NIL_PAGE_NO_GG00 != m_LeftMostIdNextLowerLevel )
        {
            m_Node.bd600GetNode (m_LeftMostIdNextLowerLevel);

#           if COMPILEMODE_MEO00 >= SLOW_MEO00 
            t01int4 (bd_index, "ToLeftMostId", m_LeftMostIdNextLowerLevel);
#           endif

            if ( e_ok == m_TrError )
            {
                bd510_SetLeftMostIdNextLowerLevel (m_Node.m_pNode);
#               if COMPILEMODE_MEO00 >= SLOW_MEO00 
                t01int4 (bd_index, "NewLeftMstID", m_LeftMostIdNextLowerLevel);
#               endif
            }
        }
    }

    if ( ! m_Node.bd600IsAccessable() || e_ok != m_TrError ) return;

    bd500_CheckStopKeyAndInterrupt (ASCENDING_GBD500);
}

/*---------------------------------------------------------------------------*/

void
00375 cbd500_Tree::bd500PrevNode ()
{
    ROUTINE_DBG_MEO00 ("bd500PrevNode");
    
    if ( // PTS 1103703 AK 30-08-1999 
        (m_Node.m_Current.curr_trans->trRteCommPtr_gg00->to_cancel) &&    
        (
        (m_select == m_Node.m_Current.curr_action) ||
        (m_verify == m_Node.m_Current.curr_action)
        )
        )                     
    {
        m_TrError = e_cancelled;
        g01opmsg (sp3p_knldiag, sp3m_info, csp3_bd_msg, csp3_n_btree,
            "BD500: Prev cancelled   " , m_Node.m_Current.curr_trans->trTaskId_gg00);
        return;
    }
    
    // PTS 1122998 UH 2003-07-09 completely rewritten (was: bd500_SetPathInfoToLeftNeighbor)

    if ( g01vtrace.vtrBdIndex_gg00 )
        bd500Print("bd500PrevNode::BEGIN");

    const tsp00_Int4       TargetLevel    = m_CurrLevel;
    const tbd_node_request NodeRequest    = m_Node.m_RequestKind;
    const tsp00_PageNo     checkId        = m_Node.bd600IsAccessable() ? m_Node.bd600GetId() : NIL_PAGE_NO_GG00;
          bool             bSearchSubNode = false;
          
    if ( m_Node.bd600IsAccessable()
         &&
         m_Node.bd600Level() != m_CurrLevel ) // only check
    {
        m_Node.bd600Release (true);
        m_TrError = e_invalid_index_structure;
        bd500Print("bd500PrevNode: m_CurrLevel is wrong");
        return;
    }

    m_Node.bd600Release (true); // if m_Node is not accessable the iterator has to stop

    // position node to left leaf or to index node to go down again
    while ( m_CurrLevel < m_RootLevel )
    {
        // 1. is left neighbour known ?
        if ( m_PathInfo[m_CurrLevel].piLeftId_bd500 != NIL_PAGE_NO_GG00 )
        {
            if ( m_PathInfo[m_CurrLevel+1].piSubNodeRecIndex_bd500 == FIRST_REC_INDEX_BD00 )
            {
                m_TrError = e_invalid_index_structure;
                bd500Print("bd500PrevNode: Wrong PathInfo1");
                return;
            }
            m_Node.bd600Move(m_PathInfo[m_CurrLevel].piLeftId_bd500);
            if ( e_ok != m_TrError )
            {
                bd500Print("bd500PrevNode::no access to left neighbor");
                return;
            }
            m_PathInfo[m_CurrLevel].piId_bd500              = m_PathInfo[m_CurrLevel].piLeftId_bd500;
            m_PathInfo[m_CurrLevel].piLeftId_bd500          = NIL_PAGE_NO_GG00;
            m_PathInfo[m_CurrLevel].piMaxRecIndex_bd500     = m_Node.bd600MaxRecIndex();
            m_PathInfo[m_CurrLevel].piSubNodeRecIndex_bd500 = m_Node.bd600MaxRecIndex();
            m_PathInfo[m_CurrLevel].piLeafCount_bd500       = m_Node.bd600LeafCount(m_Node.bd600MaxRecIndex(),m_Node.bd600MaxRecIndex());
            --(m_PathInfo[m_CurrLevel+1].piSubNodeRecIndex_bd500); // update index above
            if ( TargetLevel == m_CurrLevel )
            {
                // the simplest way - the left neighbor node is known
                if ( checkId != NIL_PAGE_NO_GG00
                     &&
                     checkId != m_Node.bd600RightNeighbor() )
                {
                    m_TrError = e_invalid_index_structure;
                    bd500Print("bd500PrevNode: Wrong PathInfo2");
                    return;
                }
                if ( g01vtrace.vtrBdIndex_gg00 )
                    bd500Print("bd500PrevNode: FOUND simple");
                return;
            }
            bSearchSubNode = true;
            break;
        }

        // invalidate the pathinfo for this level because we must search for a new way
        m_PathInfo[m_CurrLevel].piId_bd500              = NIL_PAGE_NO_GG00;
        m_PathInfo[m_CurrLevel].piLeftId_bd500          = NIL_PAGE_NO_GG00;
        m_PathInfo[m_CurrLevel].piSubNodeRecIndex_bd500 = NIL_RECINDEX_BD00;
        m_PathInfo[m_CurrLevel].piMaxRecIndex_bd500     = NIL_RECINDEX_BD00;
        m_PathInfo[m_CurrLevel].piLeafCount_bd500       = -1;

        // 2. try to use the next higher index node

        ++m_CurrLevel;

        if ( m_PathInfo[m_CurrLevel].piSubNodeRecIndex_bd500 > FIRST_REC_INDEX_BD00 )
        {
            // This index node can be used to find the leaf again.
            --(m_PathInfo[m_CurrLevel].piSubNodeRecIndex_bd500);
            m_Node.bd600Move(m_PathInfo[m_CurrLevel].piId_bd500);
            bSearchSubNode = true;
            break;
        }
    }//endwhile

    if ( ! bSearchSubNode )
    {
        if ( g01vtrace.vtrBdIndex_gg00 )
            bd500Print("bd500PrevNode: NOT FOUND");
        return; // the loop above did not find any left neighbor
    }

    /* Find leaf node again from found index node */

    // Start with determined Separator
    bd500_NextSubNode (m_PathInfo[m_CurrLevel].piSubNodeRecIndex_bd500, TargetLevel, NodeRequest);
    if ( e_ok != m_TrError )
    {
        bd500Print("bd500PrevNode:no access to subnode1");
        return;
    }
    
    while ( m_CurrLevel > TargetLevel )
    {
        bd500_NextSubNode (m_Node.bd600MaxRecIndex(), TargetLevel, NodeRequest);
        if ( e_ok != m_TrError )
        {
            bd500Print("bd500PrevNode:no access to subnode2");
            return;
        }
    }
    if ( m_CurrLevel != TargetLevel )
    {
        m_TrError = e_invalid_index_structure;
        bd500Print("bd500PrevNode: target level not reached");
        return;
    }
    
    if ( checkId != NIL_PAGE_NO_GG00
         &&
         checkId != m_Node.bd600RightNeighbor() )
    {
        m_TrError = e_invalid_index_structure;
        bd500Print("bd500PrevNode: Wrong PathInfo3");
        return;
    }
    
    if ( g01vtrace.vtrBdIndex_gg00 )
        bd500Print("bd500PrevNode::FOUND complex");

    if ( m_select == m_Node.m_Current.curr_action
         &&
         m_Node.bd600IsAccessable() )
        bd500_CheckStopKeyAndInterrupt (! ASCENDING_GBD500);
}

/*---------------------------------------------------------------------------*/

tsp00_PageNo
00533 cbd500_Tree::bd500LeftNeighborId (cbd600_Node &auxNode) const
{
    // This uses the m_PathInfo, but does not change it.
    
    ROUTINE_DBG_MEO00 ("bd500LeftNeighborId");

    // PTS 1122998 UH 2003-07-09 new

    if ( g01vtrace.vtrBdIndex_gg00 )
        bd500Print("bd500LeftNeighborId::BEGIN");

    if ( auxNode.bd600IsAccessable() )
    {
        auxNode.bd600TrError() = e_not_implemented;
        return NIL_PAGE_NO_GG00;
    }
    
    if ( m_CurrLevel >= m_RootLevel )
        return NIL_PAGE_NO_GG00;

    const tsp00_Int4   TargetLevel = m_CurrLevel;
    const tsp00_PageNo checkId     = m_PathInfo[m_CurrLevel].piId_bd500;

      tsp00_Int4   CurrentLevel        = m_CurrLevel;
    tsp00_Int4   CurrentSubNodeIndex = NIL_REC_INDEX;
    tsp00_PageNo IndexNodeId         = NIL_PAGE_NO_GG00;

          
    // position node to left leaf or to index node to go down again

    while ( CurrentLevel < m_RootLevel
            &&
            IndexNodeId == NIL_PAGE_NO_GG00 )
    {
        if ( m_PathInfo[CurrentLevel].piLeftId_bd500 != NIL_PAGE_NO_GG00 )
        {
            // left neighbor is known so take this one
            // the rightmost separator must be used
            // this is indeicated by NIL_REC_INDEX
            IndexNodeId = m_PathInfo[CurrentLevel].piLeftId_bd500;

            if ( nr_for_read == m_Node.m_RequestKind
                 &&
                 m_PathInfo[CurrentLevel+1].piSubNodeRecIndex_bd500 == FIRST_REC_INDEX_BD00 )
            {
                // if for_update this may not be valid because of overflow or underflow handling
                auxNode.bd600TrError() = e_invalid_index_structure;
                bd500Print("bd500LeftNeighborId: Wrong PathInfo1");
                return NIL_PAGE_NO_GG00;
            }
            break;
        }
        ++CurrentLevel;
        if ( m_PathInfo[CurrentLevel].piSubNodeRecIndex_bd500 > FIRST_REC_INDEX_BD00 )
        {
            // on the next higher level the index node can be used to go down again
            // the left subnode is used
            IndexNodeId         = m_PathInfo[CurrentLevel].piId_bd500;
            CurrentSubNodeIndex = m_PathInfo[CurrentLevel].piSubNodeRecIndex_bd500-1;
            break;
        }
    }

    if ( NIL_PAGE_NO_GG00 == IndexNodeId )
    {
        if ( g01vtrace.vtrBdIndex_gg00 )
            bd500Print("bd500LeftNeighborId: NOT FOUND");
        return NIL_PAGE_NO_GG00; // the loop above did not find any left neighbor
    }

    if ( CurrentLevel == TargetLevel )
    {
        if ( g01vtrace.vtrBdIndex_gg00 )
            bd500Print("bd500LeftNeighborId: FOUND simple");
        return IndexNodeId;
    }
    
    // The CurrentLevel is above the TargetLevel so go down over the rightmost separator.

    auxNode.bd600Move(IndexNodeId);
    if ( e_ok != auxNode.bd600TrError() )
    {
        auxNode.bd600Release (false); // release aux node if necessary
        bd500Print("bd500LeftNeighborId: no access to index node");
        return NIL_PAGE_NO_GG00;
    }
    
    if ( CurrentSubNodeIndex == NIL_REC_INDEX )
        CurrentSubNodeIndex = auxNode.bd600MaxRecIndex();
        
    /* Find leaf node again from found index node */

    while ( CurrentLevel > TargetLevel )
    {
        auxNode.bd600Move(auxNode.bd600RecPtr(CurrentSubNodeIndex)->recPno_gg00().gg90GetInt4());
        if ( e_ok != auxNode.bd600TrError() )
        {
            auxNode.bd600Release (false); // release aux node
            bd500Print("bd500LeftNeighborId:no access to subnode1");
            return NIL_PAGE_NO_GG00;
        }
        --CurrentLevel;
        CurrentSubNodeIndex = auxNode.bd600MaxRecIndex();
    }
    if ( CurrentLevel != TargetLevel )
    {
        auxNode.bd600Release (false); // release aux node
        auxNode.bd600TrError() = e_invalid_index_structure;
        bd500Print("bd500LeftNeighborId: target level not reached");
        return NIL_PAGE_NO_GG00;
    }
    
    if ( checkId != NIL_PAGE_NO_GG00
         &&
         auxNode.bd600IsAccessable()
         &&
         checkId != auxNode.bd600RightNeighbor() )
    {
        auxNode.bd600TrError() = e_invalid_index_structure;
        bd500Print("bd500LeftNeighborId: Wrong PathInfo3");
        return NIL_PAGE_NO_GG00;
    }
    
    if ( g01vtrace.vtrBdIndex_gg00 )
        bd500Print("bd500LeftNeighborId::FOUND complex");

    const tsp00_PageNo leftId = auxNode.bd600GetId();
    auxNode.bd600Release (false); // release aux node
    return leftId;
}

/*---------------------------------------------------------------------------*/

void
00667 cbd500_Tree::bd500Break( 
    const tsp00_Int4    recIndex,
    const tsp00_KeyPtr  rePositionKey,      
    tsp00_Int2          &rePositionKeyLen ) 
{
    ROUTINE_DBG_MEO00 ("bd500Break");

    const tgg00_RecPtr pRec = m_Node.bd600RecPtr( recIndex );

    gg05KeyAssign (REINTERPRET_CAST(tsp00_KeyPtr,pRec)+cgg_rec_key_offset, pRec->recKeyLen_gg00(),
        rePositionKey, rePositionKeyLen, m_TrError);
    if ( e_ok != m_TrError ) return;

#   if COMPILEMODE_MEO00 >= SLOW_MEO00 
    g01opmsg (sp3p_knldiag, sp3m_info, csp3_bd_msg, csp3_n_btree,
          "BD500: Interrupting  Id:" , m_Node.m_pNode->nd_id() );
#   endif

    m_Node.bd600Release(IS_CHANGED_BD600);
    m_Node.m_Current.bd300UnlockTree();
}

/*---------------------------------------------------------------------------*/

void cbd500_Tree::bd500_WriteTrace (const char      *Line,
                                    tsp00_Int4       LineLen) const
{
    Trace_CommonEntry( *(m_Node.m_Current.curr_trans),
                       tgg00_Debug::fromConst(ta), tgg00_VtraceType::fromConst(td_none), 
                       reinterpret_cast< const tsp00_Byte*> ( Line ), LineLen );
}

/*---------------------------------------------------------------------------*/

void
00702 cbd500_Tree::bd500Print (const char* Title) const
{
    const tsp00_Int4 LineSize = sizeof(tsp00_Line)+1;
    char             Line [LineSize ];
//  tsp00_Int4       LineLen;

    if ( NULL != Title )
        bd500_WriteTrace (Title, (tsp00_Int4)strlen(Title));

    if ( m_Node.m_pNode != NULL )
    {
        if ( pt2InvSubtree_egg00 == m_Node.m_pNode->nd_pt2() )
            bd500_WriteTrace ("*** InvTree (Sub) ***", 21);
        else
            bd500_WriteTrace ("*** InvTree       ***", 21);
    }
    else
        bd500_WriteTrace ("*** Tree          ***", 21);

    sp77sprintf (Line, LineSize, "* CurrLevel %d, CurrNode %d, LeftMostIdNLL %d",
                 m_CurrLevel, (m_Node.m_pNode!=NULL)?m_Node.m_pNode->nd_id():NIL_PAGE_NO_GG00,
                 m_LeftMostIdNextLowerLevel);
    bd500_WriteTrace (Line, (tsp00_Int4)strlen(Line));

    bd500_WriteTrace ("*-------+--------------+------------+----------+----------+------------+", 72);
    bd500_WriteTrace ("* Level | LeftNeighbor |       Node | RecIndex | MxRecIdx |  LeafCount |", 72);
    for (tsp00_Int4 Level = m_RootLevel; Level >= LEAF_LEVEL_BD00; --Level)
    {
        sp77sprintf (Line, LineSize, "* %5d | %10d   | %10d | %8d | %8d | %10d |",
                     Level, m_PathInfo [Level].piLeftId_bd500, m_PathInfo [Level].piId_bd500,
                     m_PathInfo [Level].piSubNodeRecIndex_bd500,
                     m_PathInfo [Level].piMaxRecIndex_bd500, m_PathInfo [Level].piLeafCount_bd500);
        bd500_WriteTrace (Line, (tsp00_Int4)strlen(Line));
    }
    bd500_WriteTrace ("*-------+--------------+------------+----------+----------+------------+", 72);
}


/*---------------------------------------------------------------------------*/

tsp00_PageNo
00743 cbd500_Tree::bd500LeftNeighbor ()
{
    ROUTINE_DBG_MEO00 ("bd500LeftNeighbor");

    if ( ! m_Node.bd600IsAccessable() )
        return NIL_PAGE_NO_GG00;

    cbd600_Node auxNode(m_Node.m_Current, m_Node.m_RequestKind);

    return bd500LeftNeighborId(auxNode);
}

/*---------------------------------------------------------------------------*/

void
00758 cbd500_Tree::bd500SetToNodeOnEdge (bool leftEdge, tsp00_Int2 level)
{
    ROUTINE_DBG_MEO00 ("bd500SetToNodeOnEdge");

    const tsp00_Int2 targetLevel = level < 0 || level > MAX_LEVEL_BD00
                                   ? LEAF_LEVEL_BD00 : level;

    bd500_Init     (); // ReInitialize the Tree 
    bd500SetToRoot (); //
    if ( e_ok != m_TrError ) return;

    tbd_node_request TargetRequest = m_Node.m_RequestKind;

    while ( m_CurrLevel > targetLevel)
    {
        if ( leftEdge )
            bd500_NextSubNode (FIRST_REC_INDEX_BD00, targetLevel, TargetRequest);
        else
            bd500_NextSubNode (m_Node.bd600MaxRecIndex(), targetLevel, TargetRequest);
        if ( e_ok != m_TrError ) break;
    }
}

/*---------------------------------------------------------------------------*/

void
cbd500_Tree::bd500_InvalidIndexStructure (tsp00_Int4     MsgNo,
                                          const char*  MsgText,
                                          tsp00_Int4     BadValue)
{
    ROUTINE_DBG_MEO00 ("bd500_InvalidIndexStructure");
    m_TrError = e_invalid_index_structure;
    bd500Print ("bd500_InvalidIndexStructure");
    g01abort (MsgNo, csp3_n_btree, MsgText, BadValue);
}

/*---------------------------------------------------------------------------*/

void
cbd500_SubTree::bd500_CheckStopKeyAndInterrupt (bool bAscending)
{
    if ( m_Node.bd600IsLeaf() )
    {
        if ( bd500_BeyondStopKey(bAscending) )
        {
#           if COMPILEMODE_MEO00 >= SLOW_MEO00 
            t01int4 (bd_index, "BeyondStp ID", m_Node.m_pNode->nd_id());
#           endif
            m_Node.bd600Release (IS_CHANGED_BD600);
        }
        else
        {
            if( ! m_Interruptible ) // PTS 1129329 TS 2004-04-26
                return;
            
            if ( NULL == m_pTreeIsRequested )
                return;

            /* check if another task wants to access the tree */
            *m_pTreeIsRequested =
                ((nr_for_read == m_Node.m_Current.bd300RootRequestKind())                       &&
                (bd20IsPageRequested (m_Node.m_Current.bd300RootPtrs().np_cbptr())              ||
                bd20IsPageRequested (m_MotherTree.m_Node.m_Current.bd300RootPtrs().np_cbptr()) ||
                m_MotherTree.m_Node.bd600NodeIsRequested()));
            
            /* check if this order was canceled */
            if ( m_Node.m_Current.curr_trans->trRteCommPtr_gg00->to_cancel             && 
                ! m_Node.m_Current.curr_trans->trState_gg00.includes(tsRollback_egg00))
            {
                m_TrError = e_cancelled;
                g01opmsg (sp3p_knldiag, sp3m_info, csp3_bd_msg, csp3_n_btree,
                    "bd500_Inter: canceled   " , m_Node.m_Current.curr_trans->trTaskId_gg00);
            }
        }
    }
}

/*===========================================================================*
 *  DEFINITION OF METHODS DECLARED IN gbd500.h (CODE)                        * 
 *===========================================================================*/

/*---------------------------------------------------------------------------*/

inline void
bd500_WriteTrace (tgg00_TransContext &Trans,
                  const char         *Line,
                  tsp00_Int4          LineLen)
{
    Trace_CommonEntry( Trans, tgg00_Debug::fromConst(ta), tgg00_VtraceType::fromConst(td_none), 
                reinterpret_cast< const tsp00_Byte*> ( Line ), LineLen );
}

/*---------------------------------------------------------------------------*/

void
cbd501_IndexOrders::bd501Trace (tgg00_TransContext &Trans)
{
    const tsp00_Int4   LineSize = sizeof(tsp00_Line)+1;
    const char        *OrderKindChar = "NCAD";
    char               Line [LineSize ];
    char              *AuxLine = Line;

    sp77sprintf (Line, LineSize, "IndexOrders [%d]", iosOrderCount_bd501);
    bd500_WriteTrace (Trans, Line, (tsp00_Int4)strlen(Line));
    bd500_WriteTrace (Trans, "------------------------------", 30);
    for (tsp00_Int4 OrderNo = 0; OrderNo < iosOrderCount_bd501; ++OrderNo)
    {
        /* PTS 1108839 UH 2000-12-27 */
        cbd501_IndexOrder &CurrOrder = iosOrders_bd501[OrderNo];
        sp77sprintf (AuxLine, LineSize, "%c%d#%d%s,",
            OrderKindChar[CurrOrder.ioKind_bd501],
            CurrOrder.ioSubNodeId_bd501(),
            CurrOrder.ioLeafCount_bd501(),
            CurrOrder.ioKeyIsInvalid()?(""):("K"));
        AuxLine = Line + strlen(Line);
        // CurrOrder->ioSepKey_bd501(), CurrOrder->ioSepKeyLen_bd501
    }
    bd500_WriteTrace (Trans, Line, (tsp00_Int4)strlen(Line));
    bd500_WriteTrace (Trans, "------------------------------", 30);
}

/*===========================================================================*
 *  GLOBAL FUNCTIONS (CODE)                                                  *
 *===========================================================================*/

/*===========================================================================*
 *  LOCAL FUNCTIONS (CODE)                                                   *
 *===========================================================================*/

/*===========================================================================*
 *  END OF CODE                                                              *
 *===========================================================================*/

Generated by  Doxygen 1.6.0   Back to index