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

Kernel_Administration.cpp

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

  module      : Kernel_Administration.cpp
  special area: Kernel Management
  responsible : UweH
  created     : 2002-12-11
  last changed: 2002-12-11  12:00
  copyright:    (c) 2000-2004 SAP AG
  description : This is the handler which coordinates all
                DBM commands in the kernel. (utility)

    ========== licence begin  GPL
    Copyright (c) 2000-2004 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 "KernelCommon/Kernel_Administration.hpp"

#include "SAPDBCommon/MemoryManagement/SAPDBMem_IRawAllocator.hpp"
#include "SAPDBCommon/MemoryManagement/SAPDBMem_NewDestroy.hpp"

#include "SAPDBCommon/ErrorsAndMessages/SAPDBErr_Assertions.hpp"
#include "SAPDBCommon/Tracing/SAPDBTrace_Usage.hpp"
#include "SAPDBCommon/SAPDB_sprintf.h"

#include "RunTime/RTE_ISystem.hpp"
#include "RunTime/RTE_Message.hpp"
#include "RunTime/HotStandbyStorage/RTEHSS_KernelInterface.hpp"
#include "RunTime/HotStandbyStorage/RTEHSS_StandbyNodeList.hpp"
#include "RunTime/Configuration/RTEConf_ParameterAccessKernelInterface.hpp"

#include "KernelCommon/Kernel_Types.hpp"
#include "KernelCommon/Kernel_DateTime.hpp"
#include "KernelCommon/Kernel_Messages.hpp"
#include "KernelCommon/Kernel_Exception.hpp"
#include "KernelCommon/Kernel_Version.hpp"
#include "KernelCommon/Kernel_RestartPage.hpp"

#include "IOManager/IOMan_IDataManager.hpp"
#include "IOManager/IOMan_IDataInfo.hpp"
#include "IOManager/IOMan_IDataVolumeInfo.hpp"
#include "IOManager/IOMan_ILogVolumeInfo.hpp"
#include "IOManager/IOMan_IDataIO.hpp"
#include "IOManager/IOMan_ILogManager.hpp"
#include "IOManager/IOMan_ILogInfo.hpp"
#include "IOManager/IOMan_IDiagnose.hpp"
#include "IOManager/IOMan_BlockAddress.hpp"
#include "IOManager/IOMan_ReservedBlockAddress.hpp"

#include "Logging/Log_Types.hpp"
#include "Logging/Log_Savepoint.hpp"
#include "Logging/Log_Volume.hpp"
#include "Logging/Log_History.hpp"

#include "FrameControl/FrameCtrl_ILog.hpp"

#include "Converter/Converter_IBackUp.hpp"
#include "Converter/Converter_ICommon.hpp"
#include "Converter/Converter_IManager.hpp"
#include "Converter/Converter_IDiagnose.hpp"

#include "DataAccess/Data_Types.hpp"

#include "SQLManager/Catalog/Catalog_Interface.hpp"
#include "SQLManager/ErrorHandling/SQLManErr_Interface.hpp"

#include "ggg92.h"   // tgg92_KernelOid
#include "ggg00.h"   // tgg00_TransContext
#include "hak10.h"   // a10expand_rec
#include "hbd01.h"   // b01restart_filesystem
#include "hbd01_1.h" // b01niltree_id
#include "hbd02.h"   // b02next_record
#include "hbd03.h"   // bd03ExtractIndex
#include "hbd05.h"   // bd05ExtractBlob
#include "hbd06.h"   // b06create_errortext_file
#include "hgg01_3.h" // g01tabid
#include "hgg10.h"   // g10mv
#include "hgg17.h"   // g17basis_err_to_line
#include "hkb51.h"   // k51restart_locklist
#include "hkb57.h"   // k57save_restartrecord
#include "hkb57_1.h" // k57restartrec
#include "hkb53.h"   // k53child_trans_build
#include "hkb38.h"   // k38init_restore / k38headmaster
#include "hkb38_1.h" // k38is_on_autosave
#include "hkb90.h"   // k90send
#include "hbd14.h"   // bd14RestartFileSystemForSaveDataCold,bd14ShutdownFileSystemForSaveDataCold
#include "hbd17.h"   // bd17GetFdirRoot, bd17GetLongFdirRoot
#include "hbd20.h"   // bd20FlushDataCache
#include "heo51.h"   // voffline
#include "heo60.h"   // vtime_in_seconds
#include "heo79.h"   // vxparam_save_good_config
#include "heo58.h"   // vget_uniqe_id
#include "heo51.h"   // vgetpid
#include "hsp20.h"   // s20ch4sw
#include "heo47.h"   // sql47_ptoc
#include "heo670.h"  // vtracestack
/* --------------------------------------------------------------------------- */
/// This suspend reason is used to wait for end of redo in case of HotStandby::TakeOver
#define KERNEL_ADMIN_WAIT_FOR_TAKEOVER 250 // suspend reasons: heo00x.h
/* --------------------------------------------------------------------------- */
static inline const char *  BasisErrorToString (const tgg00_BasisError berr)
{
    return SQLManErr_Interface::GetInstance().GetErrorText(
           SQLManErr_Interface::GetInstance().
           GetReturnCode(berr,sqlm_internal));
}
/* --------------------------------------------------------------------------- */
static void PickUpAuxFilesForDestroy( tgg00_TransContext &trans )
{
    
    tgg00_FileId    auxFileId = b01niltree_id;
    
    auxFileId.fileTfn_gg00().becomes( tfnAux_egg00 );
    
    do
    {
        b01next_fdir_entry( trans, auxFileId );
        
        if( tfnAux_egg00 != auxFileId.fileTfn_gg00())
            break;

        if( e_ok == trans.trError_gg00 )
            kb90AddToPrefixDestroyFile( trans, auxFileId.fileTransId__7_gg00());
    }
    while( e_ok == trans.trError_gg00 );
    
    trans.trError_gg00 = e_ok; // ignore errors
}
/* --------------------------------------------------------------------------- */
Kernel_Administration* Kernel_Administration::m_Instance = 0;
/* --------------------------------------------------------------------------- */
00150 void Kernel_Administration::CreateInstance (SAPDBMem_IRawAllocator &allocator)
{
    SAPDBTRACE_ROUTINE_DEBUG ("Kernel_Administration::CreateInstance", Common_Trace, 5);
    SAPDBMEM_STATIC_RAW_ALLOCATE (Space, sizeof(Kernel_Administration));
    SAPDBERR_ASSERT_STATE(m_Instance == 0);
    m_Instance = new (Space) Kernel_Administration (allocator);
}
/* --------------------------------------------------------------------------- */
00158 Kernel_Administration& Kernel_Administration::Instance()
{
    SAPDBERR_ASSERT_STATE(m_Instance != 0);
    return *m_Instance;
}
/* --------------------------------------------------------------------------- */
00164 Kernel_IAdminRestartShutdown& Kernel_IAdminRestartShutdown::Instance()
{
    return Kernel_Administration::Instance();
}
/* --------------------------------------------------------------------------- */
00169 Kernel_IAdminHotStandby& Kernel_IAdminHotStandby::Instance()
{
    return Kernel_Administration::Instance();
}
/* --------------------------------------------------------------------------- */
00174 Kernel_IAdminBackupRecovery& Kernel_IAdminBackupRecovery::Instance()
{
    return Kernel_Administration::Instance();
}
/* --------------------------------------------------------------------------- */
00179 Kernel_IAdminInfo& Kernel_IAdminInfo::Instance()
{
    return Kernel_Administration::Instance();
}
/* --------------------------------------------------------------------------- */
00184 Kernel_IAdminConfig& Kernel_IAdminConfig::Instance()
{
    return Kernel_Administration::Instance();
}
/* --------------------------------------------------------------------------- */
00189 Kernel_IAdminDiagnose& Kernel_IAdminDiagnose::Instance()
{
    return Kernel_Administration::Instance();
}
/* --------------------------------------------------------------------------- */
00194 Kernel_Administration::Kernel_Administration(SAPDBMem_IRawAllocator &allocator)
: m_Allocator(UTF8("Kernel_Administration"),allocator),
  m_SpinLock((SAPDB_UTF8*)"Kernel_Administration::m_SpinLock"),
  m_CommandNo(0),
  m_State(m_SpinLock),
  m_WaitingForTakeOverReady(cgg_nil_pid),
  m_SynchronizationIsRunning(false),
  m_SynchronizationIsNeeded(false)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::Kernel_Administration", Common_Trace, 5);
    RTE_Message (Admin_Exception(__CONTEXT__, KERNEL_STATE_IS_ADMIN));
}
/* --------------------------------------------------------------------------- */
00207 void Kernel_Administration::Activate (tgg00_TransContext &trans)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::Activate", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateRedo, m_State);
    
    if ( ! state.EnterCriticalState() )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Redo"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    RestartIntern (trans, restartKindActivation, Kernel_Date(), Kernel_Time(), Log_IOSequenceNo());
}
/* --------------------------------------------------------------------------- */
00223 void Kernel_Administration::Restart (tgg00_TransContext &trans,
                                     Kernel_Date         untildate,
                                     Kernel_Time         untiltime,
                                     Log_IOSequenceNo    untilIOSequence)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::Restart", Common_Trace, 5);

    if ( m_State.IsOnline() )
    {
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    Kernel_StateScope stateRedo (Kernel_State::criticalStateRedo, m_State);
    
    if ( ! stateRedo.EnterCriticalState() )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Redo"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    RestartIntern (trans, restartKindNormal, untildate, untiltime, untilIOSequence);
}

/* --------------------------------------------------------------------------- */
00249 bool Kernel_Administration::IsMasterAlive(SAPDBErr_MessageList &errlist)
{
    // PTS 1127374 UH 2004-01-23 new
    RTE_Nodename                    masterNodeName;
    RTEHSS_KernelInterface::Mapping dummyOtherMapping;
    
    if ( ! GetMaster (masterNodeName, errlist) )
        return true; // for safety - to not destroy the log assume master is alive
        
    if ( RTEHSS_KernelInterface::Instance().
         SendInfoStandby(masterNodeName, dummyOtherMapping, errlist) )
        return true;
    errlist.ClearMessageList(); // its not interesting
    return false;
}
    
/* --------------------------------------------------------------------------- */
00266 bool Kernel_Administration::RestartPrepare (tgg00_TransContext   &trans,
                                            RestartKind           restartkind,
                                            SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RestartPrepare", Common_Trace, 5);

    RTEHSS_KernelInterface &hss = RTEHSS_KernelInterface::Instance();

    if ( ! RestartFilesystem(trans) )
    {
        errlist = Admin_Exception(__CONTEXT__, KERNEL_RESTART_FILESYSTEM_FAILED,
                                  BasisErrorToString(trans.trError_gg00));
        return false;
    }
    
    if ( ! CheckAndUpdateActivationState (trans, restartkind, errlist) )
        return false;
    
    k51restart_locklist(trans);
    
    if ( trans.trError_gg00 != e_ok )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_RESTART_LOCKLIST_FAILED,
                                  BasisErrorToString(trans.trError_gg00));
        return false;
    }

    if ( restartKindStandby != restartkind )
    {
        // Check if restart is allowed for a standby
        // Is the given master alive ? If so we must not restart.
        if ( m_State.IsStandby() && IsMasterAlive(errlist) ) // PTS 1127374 UH 2004-01-23
        {
            errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Restart","master is alive");
            return false;
        }
    }
    else
    {
        // Register this stand by and prevent the master from overwriting our savepoint offset
        // It may be that our savepoint offset is already overwritten since the establishment
        // of the mirror but this will be detected later. (LogDataIncompatible)
        SAPDB_UInt4 auxOffset;
        if ( ! hss.SendRegisterStandby( k57restartrec->rstLastSavept_kb00().svpStartOffset_kb00,
                                        auxOffset, errlist ) )
            return false;
    
        RTE_Message(Admin_Exception(__CONTEXT__, KERNEL_STANDBY_REGISTER, SAPDB_ToString(auxOffset)));
        
        m_LastSyncOffset = auxOffset;
        
        // db_state is now changed from ADMIN to STANDBY
        if ( ! hss.SetKernelStateToStandby(errlist) )
            return false;
    }

    if ( ! RestartLog (trans.trTaskId_gg00, restartkind, errlist) )
        return false;

    if ( ! CheckAndResetReadOnlyState(trans) )
    {
        errlist = Admin_Exception(__CONTEXT__, KERNEL_RESET_READONLY_FAILED,
                                  BasisErrorToString(trans.trError_gg00));
        return false;
    }
    
    return true;
}

/* --------------------------------------------------------------------------- */
00336 void Kernel_Administration::RestartIntern (tgg00_TransContext &trans,
                                           RestartKind         restartkind,
                                           Kernel_Date         untildate,
                                           Kernel_Time         untiltime,
                                           Log_IOSequenceNo    untilIOSequence)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RestartIntern", Common_Trace, 5);

    tsp00_TaskId        &taskid         = trans.trTaskId_gg00;
    tgg00_BasisError    &trError        = trans.trError_gg00;

    if ( restartKindStandby != restartkind )
    {
        SAPDBErr_MessageList errlist;
        if ( ! RestartPrepare (trans, restartkind, errlist) )
        {
            RTE_Message(errlist);
            Offline (trError);
        }
    }

    if ( restartKindRestoreLog != restartkind )
    {
        Rst_RedoKind redokind;
        
        switch ( restartkind )
        {
            case restartKindNormal:
            case restartKindActivation:
                redokind = Rst_RedoKindNormal;
                break;
            case restartKindRestoreLog:
                redokind = Rst_RedoKindRestoreLog;
                break;
            case restartKindStandby:
                redokind = Rst_RedoKindStandby;
                break;
            default:
                redokind = Rst_RedoKindNormal;
                break;
        }

        if ( restartKindStandby != restartkind )
                  m_LastSyncOffset.Invalidate();

        if ( ! RedoLog (trans, redokind, m_LastSyncOffset, untildate, untiltime, untilIOSequence) )
            Offline (trError);
    }
    
    // The persistent data is now transactional consistent

    if ( g01is_livecache()
         &&
         ! Log_History::GetInstance().CheckAndStartMigrationIfNeeded(trans) )
        Offline(trError);

    if ( g01is_livecache()
         &&
         restartKindRestoreLog != restartkind  )
    {
        // for livecache delay the reading of the objects
        bd01CreateCoordinator (trans, m_get_object);
        if ( trError != e_ok )
            Offline (trError);
    }
     
    tsp00_ErrText      xp_errortext;
    tsp00_XpReturnCode xp_return;
    vxparam_save_good_config (restartKindActivation != restartkind, xp_errortext, xp_return);
    if ( xp_return != xp_ok )
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_SAVE_GOOD_FAILED,
                                      SAPDB_ToString(xp_return), xp_errortext));

    if ( restartKindRestoreLog != restartkind )
    {
        RTE_Message (Admin_Exception(__CONTEXT__, KERNEL_REDO_LOG_FINISHED));
    }
}
/* --------------------------------------------------------------------------- */
00415 void Kernel_Administration::Shutdown (tgg00_TransContext &trans)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::Shutdown", Common_Trace, 5);
    
    if ( ! m_State.IsOnline()  )
    {
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    // PTS 1128407 UH 2004-03-10 added synchronization for multiple shutdown requests
    Kernel_StateScope stateRedo (Kernel_State::criticalStateRedo, m_State);
    if ( ! stateRedo.EnterCriticalState() )
        // more then one shutdown request occured but no error is wanted
        return;

    const tgg00_BasisError auxError = trans.trError_gg00;
    trans.trError_gg00 = e_ok;

    /* PTS 1128407 UH 2004-03-10 do not check autosave here
       it is disabled by Log_Savepoint::Coordinator()
    if ( k38is_on_autosave (trans, false) )
    {
        trans.trError_gg00 = e_autosave_running;
        return;
    }
    trans.trError_gg00 = e_ok;
    */

    Log_SavepointManager.StartSavepointAndWait (trans, Log_SVPReasonShutdown);
    // it should not be possible to reach this code
    Offline (auxError);
}
/* --------------------------------------------------------------------------- */
00449 void Kernel_Administration::Offline (SAPDB_Int2 error,
                                     bool       forceBacktrace)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::Offline", Common_Trace, 5);

    tsp00_Line auxLine;

    if ( error == SAPDB_MAX_INT2 )
    {
        auxLine.c2p("");
        auxLine[auxLine.length()] = 0;
    }
    else
    {
        int auxLineLength = 0;
        g17basis_err_to_line (error, auxLineLength, auxLine);
        auxLine[auxLineLength] = 0;
    }

    if ( forceBacktrace || Common_Trace.TracesLevel(9) )
        vtracestack();
      
    tsp00_TaskId taskid;

    vgetpid(taskid);

    if ( error != e_ok )
    {
        RTE_Message(Admin_Exception(__CONTEXT__, KERNEL_FORCE_OFFLINE, auxLine));
        voffline (taskid, shtShutdKill_esp00);
    }

    RTE_Message(Admin_Exception(__CONTEXT__, KERNEL_STATE_IS_OFFLINE));
    voffline (taskid, shtShutdNormal_esp00);
    // -------------------------
    // THE KERNEL IS OFFLINE NOW
    // -------------------------
}
/* --------------------------------------------------------------------------- */
00488 void Kernel_Administration::GetLastEntryRead(Log_EntryInfo &entryInfo)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::GetLastEntryRead", Common_Trace, 5);
    m_RedoManager.GetLastEntryRead(entryInfo);
}
/* --------------------------------------------------------------------------- */
00494 void Kernel_Administration::ReadLogAndCreateRedoFiles(tgg00_TransContext &readerTrans)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::ReadLogAndCreateRedoFiles", Common_Trace, 5);
    m_RedoManager.ReadLogAndCreateRedoFiles(readerTrans);
}
/* --------------------------------------------------------------------------- */
00500 bool Kernel_Administration::RestartFilesystem (tgg00_TransContext &trans,
                                               const bool         incrementConverterVersion)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RestartFilesystem", Common_Trace, 5);

    b01restart_filesystem (trans, incrementConverterVersion);
    if ( trans.trError_gg00 != e_ok )
        return false;
        
    b01filestate (trans, g01tabid.sys1_cat);
    if ( trans.trError_gg00 != e_ok )
        return false;

    b01filestate (trans, g01tabid.sys2_cat);
    if ( trans.trError_gg00 != e_ok )
        return false;

    return true;
}
/* --------------------------------------------------------------------------- */
00520 bool Kernel_Administration::CheckAndUpdateActivationState (tgg00_TransContext   &trans,
                                                           RestartKind           restartkind,
                                                           SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::CheckAndUpdateActivationState", Common_Trace, 5);

    SAPDBERR_ASSERT_STATE(m_State.IsRedo());

    const tkb00_ConfigPhase state = k57restartrec->rstConfigPhase_kb00();

    if ( restartKindActivation == restartkind )
    {
        if ( state != cfpConfigured_ekb00 )
        {
            errlist = Admin_Exception(__CONTEXT__,KERNEL_ALREADY_ACTIVATED);
            return false;
        }

        k57restartrec->rstConfigPhase_kb00().becomes(cfpNone_ekb00);
        FlushRestartRecord (trans.trTaskId_gg00);
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_SWITCHED_TO_ACTIVATED));
        return true;
    }
    
    if ( restartKindRestoreLog == restartkind
         &&
         state == cfpConfigured_ekb00 )
    {
        // restore log without initial data recovery
        k57restartrec->rstConfigPhase_kb00().becomes(cfpNone_ekb00);
        FlushRestartRecord (trans.trTaskId_gg00);
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_SWITCHED_TO_ACTIVATED2));
        // if the restore log fails
        // it must be called init_config again
        return true;
    }

    if ( state == cfpNone_ekb00 )
        return true; // normal restart

    if ( state == cfpConfigured_ekb00 )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_ACTIVATE_IS_MISSING);
        return false;
    }

    errlist =  Admin_Exception( __CONTEXT__,
                                KERNEL_UNKNOWN_ACTIVATION_STATE,
                                SAPDB_ToString(state));
    return false;
}
/* --------------------------------------------------------------------------- */
00572 bool Kernel_Administration::RestartLog (tsp00_TaskId          taskid,
                                        RestartKind           reason,
                                        SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RestartLog", Common_Trace, 5);

    Log_Volume::RestartType logRestartType;
    
    switch ( reason )
    {
        case restartKindNormal:
        case restartKindActivation:
            logRestartType = Log_Volume::RestartNormal;
            break;
        case restartKindRestoreLog:
            logRestartType = Log_Volume::RestartForRestore;
            break;
        case restartKindStandby:
            logRestartType = Log_Volume::RestartForStandby;
            break;
        default:
            return false;
    }
    
    const Log_ErrorState logRestartResult = Log_Volume::Instance().Restart (taskid, logRestartType);
    if ( logRestartResult == Log_Ok )
        return true;

    Log_Volume::Instance().WriteDeviceParametersToTrace();
    errlist = Admin_Exception(__CONTEXT__, KERNEL_INIT_LOG_FAILED,
                              (restartKindRestoreLog==reason?"restore log":"restart"),
                              Log_ErrorStateStrings[logRestartResult]);
    return false;
}
/* --------------------------------------------------------------------------- */
00607 bool Kernel_Administration::CheckAndResetReadOnlyState (tgg00_TransContext &trans)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::CheckAndResetReadOnlyState", Common_Trace, 5);
    SAPDBERR_ASSERT_STATE(m_State.IsRedo());
        
    if ( k57restartrec->rstSetEndReadOnly_kb00() != 0 )
        b01end_read_only (trans);

    return trans.trError_gg00 == e_ok;
}
/* --------------------------------------------------------------------------- */
00618 bool Kernel_Administration::RedoLog (tgg00_TransContext &trans,
                                     Rst_RedoKind        redokind,
                                     Log_RawDeviceOffset lastRedoOffset,
                                     Kernel_Date         untildate,
                                     Kernel_Time         untiltime,
                                     Log_IOSequenceNo    untilIOSequence)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RedoLog", Common_Trace, 5);

    if ( ! m_State.IsRedo()  )
        Offline (e_dbm_command_not_possible);

    if ( untildate.IsValid() && untiltime.IsValid()
         &&                                     // PTS 1120507 mb 2002-02-10
         Kernel_DateTime::OlderThan (untildate, untiltime,
                                     k57restartrec->rstLastSavept_kb00().svpStartEntryDate_kb00,
                                     k57restartrec->rstLastSavept_kb00().svpStartEntryTime_kb00) )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_UNTILDATE_TOO_OLD,
                                     SAPDB_ToString(k57restartrec->rstLastSavept_kb00().svpStartEntryDate_kb00),
                                     SAPDB_ToString(k57restartrec->rstLastSavept_kb00().svpStartEntryTime_kb00)));
        trans.trError_gg00 = e_incompatible_log;
        return false;
    }

    m_RedoManager.RedoLog ( trans, lastRedoOffset, untildate, untiltime, untilIOSequence, redokind );

    if ( trans.trError_gg00 != e_ok )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_REDO_ERROR,SAPDB_ToString(trans.trError_gg00)));
        return false;
    }

    if ( IsHotStandbyConfigured()
         &&
         ! m_State.IsMaster() )
    {
        // The node which can restart must be the master and so the hostname is written to the log info page.
        // detach and attach again the log volume
        Log_Volume::Instance().CloseLogArea(trans.trTaskId_gg00);
        ForceHotStandbyRoleToMaster();
        if ( ! Log_Volume::Instance().OpenLogArea(trans.trTaskId_gg00) )
            return false;
    }
    RTE_Nodename masternodename;
    memcpy ( masternodename, RTE_ISystem::Instance().GetLocalNodeName(), sizeof(masternodename) );
    Log_Volume::Instance().SetMasterNodeName( trans.trTaskId_gg00, masternodename);

    Log_SavepointManager.StartSavepointAndWait (trans, Log_SVPReasonRestart);

    b06create_errortext_file (trans);

    if( e_ok == trans.trError_gg00 )
        PickUpAuxFilesForDestroy( trans ); // PTS 1126313 TS 2003-12-15

      if ( trans.trError_gg00 != e_ok )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_CREATE_ERROR_FILE_FAILED));
        return false;
    }

    if ( Rst_RedoKindRestoreLog == redokind )
        RTE_Message (Admin_Exception(__CONTEXT__, KERNEL_REDO_LOG_FINISHED));

    return true;
}
/* --------------------------------------------------------------------------- */
00685 void Kernel_Administration::AbortRedoLog (const tgg00_BasisError abortReason)
{
    m_RedoManager.AbortRedo(abortReason);
}
/* --------------------------------------------------------------------------- */
00690 void Kernel_Administration::SetNewDBIdentifierIfHistoryLost (tsp00_TaskId taskid)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::SetNewDBIdentifierIfHistoryLost", Common_Trace, 5);

    if ( m_State.IsStandby() )
        return; // This is not allowed for standbys
        
    Log_Volume &log = Log_Volume::Instance();

    if ( ! log.PrepareLogInfoPageForRead(taskid) )
        return; // Error on accessing the log are ignored.
        
    if ( (log.GetDeviceState() == Log_DeviceStateCleared
          ||
          log.GetDeviceState() == Log_DeviceStatePartiallyCleared
          ||
          log.GetDeviceState() == Log_DeviceStateHistoryLost)
         &&
         log.IsDevspaceEnabled(taskid) )
    {
        // a new backuphistory has began
        tsp00_Line            newDBIdent;      
        vget_uniqe_id  (newDBIdent);
        tkb00_SaveptParam &SaveptParam = k57restartrec->rstLastSavept_kb00();
    
        k57restartrec->rstDbIdent_kb00() = newDBIdent;
        
        if ( Log_DeviceStateCleared == log.GetDeviceState()
             &&
             m_State.IsAdmin() )
        {
            // adjust the offset if the log is empty, because the
            // first entry is written onto offset 24 in page 1
            // PTS 1116123 mb 2002-06-12
            SaveptParam.svpStartEntryType_kb00   = Log_FinishOpenTrans;
            SaveptParam.svpStartEntryOffset_kb00 = Log_Page::MinSpaceOffset();
            // PTS 1130390 mb 2004-06-29
            SaveptParam.svpIOsequence_kb00       = LOG_INITIAL_IOSEQ.RawValue();
            SaveptParam.svpStartOffset_kb00      = log.LogicalLogBeginOffset();
        }
    
        FlushRestartRecord (taskid);
        log.SetLogDBIdent (taskid, newDBIdent);
        
        SAPDB_Char newLogDBIdent[sizeof(tsp00_Line)+1];
        sql47_ptoc(newLogDBIdent, k57restartrec->rstDbIdent_kb00(), sizeof(tsp00_Line));
        RTE_Message(Log_Exception(__CONTEXT__, LOG_NEW_DBIDENT,newLogDBIdent));
    }
}
/* --------------------------------------------------------------------------- */
00740 void Kernel_Administration::SetKernelStateOnline()
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::SetKernelStateOnline", Common_Trace, 5);
    m_State.SetOnline();
    RTE_Message (Admin_Exception(__CONTEXT__, KERNEL_STATE_IS_ONLINE));
}
/* --------------------------------------------------------------------------- */
00747 void Kernel_Administration::RestoreLog (tgg00_TransContext &trans,
                                        tgg00_MessBlock    &restoreDescription)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RestoreLog", Common_Trace, 5);

    if ( m_State.IsStandby() )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Log recovery","StandByRole"));
        trans.trError_gg00 = e_hotstandby_wrong_role;
        return;
    }
    if ( ! m_State.EnterCriticalState(Kernel_State::criticalStateBackup) )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Backup"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    if ( ! m_State.EnterCriticalState(Kernel_State::criticalStateRedo) )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Redo"));
        m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    trans.trError_gg00 = e_ok;

    const bool onlyCheck = IsCheckBackupConfigured();
    
    if ( ! onlyCheck )
    {
        // Until dat and time is given to the backup/restore component
        // which calls RedoLog later.
        RestartIntern (trans, restartKindRestoreLog, Kernel_Date(), Kernel_Time(), Log_IOSequenceNo());
        if ( trans.trError_gg00 != e_ok )
            Offline (trans.trError_gg00);
    }

    k38init_restore (restoreDescription, onlyCheck);
    if ( trans.trError_gg00 != e_ok )
        Offline (trans.trError_gg00);

    k38headmaster (restoreDescription);
    if (    trans.trError_gg00 != e_ok
         && trans.trError_gg00 != e_new_hostfile_required 
         && trans.trError_gg00 != e_wrong_hostfile
         && trans.trError_gg00 != e_hostfile_error // PTS 1124787 mb 2003-10-23
         && trans.trError_gg00 != e_incompatible_log 
         && trans.trError_gg00 != e_invalid_label 
         && trans.trError_gg00 != e_save_skipped )
    {
        // S H U T   D O W N   T H E   K E R N E L   F O R   S E V E R E   E R R O R S
        Offline (trans.trError_gg00);
    }

    if ( e_new_hostfile_required == trans.trError_gg00 )
        return;
        
    if ( trans.trError_gg00 != e_ok )
    {
        tgg00_BasisError       logError = trans.trError_gg00;
        b01shutdown_filesystem(trans);
        trans.trError_gg00 = logError;
    }
    m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
    m_State.LeaveCriticalState(Kernel_State::criticalStateRedo);
}
/* --------------------------------------------------------------------------- */
00815 void Kernel_Administration::ClearLog (tsp00_TaskId taskid)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::ClearLog", Common_Trace, 5);

    Kernel_StateScope state1 (Kernel_State::criticalStateRedo, m_State);
    Kernel_StateScope state2 (Kernel_State::criticalStateBackup, m_State);
    Kernel_StateScope state3 (Kernel_State::criticalStateConfiguration, m_State);

    if ( ! m_State.IsAdmin()
         ||
         ! state1.EnterCriticalState()
         ||
         ! state2.EnterCriticalState()
         ||
         ! state3.EnterCriticalState() )
        Offline (e_dbm_command_not_possible);

   ClearLogArea( taskid );
}
/* --------------------------------------------------------------------------- */
00835 void Kernel_Administration::ClearLogArea (tsp00_TaskId taskid)
{
    ReadRestartRecord (taskid);
    
    Log_ErrorState result = Log_Volume::Instance().Restart (taskid, Log_Volume::RestartForFormat);
    
    if (result != Log_Ok)
    {
        Log_Volume::Instance().WriteDeviceParametersToTrace();
        RTE_Crash( Admin_Exception(__CONTEXT__, KERNEL_CLEAR_LOG_ERROR,
                                    Log_ErrorStateStrings[result]) );
    }

    Log_Volume::Instance().ClearLogComplete(taskid);
    
    Log_Volume::Instance().CloseLogArea(taskid);
}
/* --------------------------------------------------------------------------- */
00853 bool Kernel_Administration::DenyBackupAndRestore()
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::DenyBackupAndRestore", Common_Trace, 5);
    return m_State.EnterCriticalState(Kernel_State::criticalStateBackup);
}
/* --------------------------------------------------------------------------- */
00859 void Kernel_Administration::PermitBackupAndRestore()
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::PermitBackupAndRestore", Common_Trace, 5);
    m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
}
/* --------------------------------------------------------------------------- */
00865 void Kernel_Administration::BackupData (tgg00_TransContext &trans,
                                        DataBackupType      backupType,
                                        tgg00_MessBlock    &backupDescription)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::BackupData", Common_Trace, 5);

    if ( ! Log_SavepointManager.IsSavepointAllowed() ) // PTS 1121281 UH 2003-03-24
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Data Backup","No savepoint allowed"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    if ( ! m_State.EnterCriticalState(Kernel_State::criticalStateBackup) )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Backup"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    trans.trError_gg00 = e_ok; 

    if ( m_State.IsOnline() )
    {
        vbegexcl (trans.trTaskId_gg00, g08savepoint);
        k57restartrec->rstDataBackupCnt_kb00()++;
        FlushRestartRecord (trans.trTaskId_gg00);
        vendexcl (trans.trTaskId_gg00, g08savepoint);
        Log_SavepointManager.StartSavepointAndWait (trans,
                                                    (backupComplete == backupType
                                                        ? Log_SVPReasonSaveData
                                                        : Log_SVPReasonSavePages));
        if ( trans.trError_gg00 != e_ok )
        {
            k38history_error_write ( trans.trTaskId_gg00,
                                     backupDescription.mb_qual()->msave_restore().sripUtilCmdId_gg00,
                                     backupDescription.mb_type(),
                                     backupDescription.mb_type2(),
                                     false,
                                     k57restartrec->rstDataBackupCnt_kb00(),
                                     trans.trError_gg00 );
            Converter_IBackUp::Instance().EndSave (trans.trTaskId_gg00, false);
            m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
            return;
        }
    }
    else
    {
        bd14RestartFileSystemForSaveDataCold (trans.trTaskId_gg00, trans.trError_gg00);
        if ( trans.trError_gg00 != e_ok )
        {
            bd14ShutdownFileSystemAfterSaveDataCold (trans.trTaskId_gg00);
            m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
            return;
        }
        if ( backupComplete == backupType )
            k57restartrec->rstLastSaveDataSuccessful_kb00() = false;
        k57restartrec->rstDataBackupCnt_kb00()++;
        FlushRestartRecord (trans.trTaskId_gg00);

        if ( backupIncrementell == backupType )
            Converter_IBackUp::Instance().BeginSavePages (trans.trTaskId_gg00);
        else
        {
           IOMan_PackedBlockAddress  packedRootBlockAddr;

            Converter_IBackUp::Instance().BeginSaveData (trans.trTaskId_gg00);
           
            if (! Converter_IManager::Instance().FlushSerialForColdSave (
                trans.trTaskId_gg00, packedRootBlockAddr))
            {
                trans.trError_gg00 = e_shutdown;
                bd14ShutdownFileSystemAfterSaveDataCold (trans.trTaskId_gg00);
                m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
                return;
            }
            k57restartrec->rstConfigParam_kb00().crConvRootBlockAddr_kb00 = packedRootBlockAddr;

            FlushRestartRecord (trans.trTaskId_gg00);
        }
    }

    k38init_save (backupDescription, m_State.IsAdmin(), false, false);

    if ( trans.trError_gg00 != e_ok )
    {
        if ( m_State.IsOnline() )
            Converter_IBackUp::Instance().EndSave (trans.trTaskId_gg00, false);
        else
            bd14ShutdownFileSystemAfterSaveDataCold (trans.trTaskId_gg00);
        m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
        return;
    }

    k38headmaster (backupDescription);
        
    if ( trans.trError_gg00 == e_new_hostfile_required )
        return;

    if ( trans.trError_gg00 != e_ok
         &&
         ! m_State.IsOnline() )
        bd14ShutdownFileSystemAfterSaveDataCold (trans.trTaskId_gg00);

    m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
} 
/* --------------------------------------------------------------------------- */
00971 void Kernel_Administration::BackupLog (tgg00_TransContext &trans,
                                       tgg00_MessBlock    &backupDescription)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::BackupLog", Common_Trace, 5);

    if ( m_State.IsStandby() )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Log recovery","StandByRole"));
        trans.trError_gg00 = e_hotstandby_wrong_role;
        return;
    }
    if ( ! m_State.EnterCriticalState(Kernel_State::criticalStateBackup) )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Backup"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    tgg00_SaveRestoreInputParam &backupParameter    = backupDescription.mb_qual()->msave_restore();
    tgg00_MessBufHeader          initial_mb_head    = backupDescription.mb_header();
    tgg00_SaveRestoreInputParam  inital_user_data   = backupParameter;
    bool                         firstTimeTapeCheck = true;
    bool                         completeLog        = backupParameter.sripFileVersion_gg00 < 0
                                                      &&
                                                      backupParameter.sripHostFiletypes_gg00[0] != vf_t_tape_norew;
    
    if ( m_State.IsOnline() )
    {
        if ( k38is_on_autosave (trans, false) )
        {
            trans.trError_gg00 = e_autosave_running;
            m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
            return;
        }
    }
    else
    {
        const Log_ErrorState  result = Log_Volume::Instance().RestartForSaveLogCold(trans.trTaskId_gg00);

        if ( result != Log_Ok )
        {
            Log_Volume::Instance().WriteDeviceParametersToTrace();
            RTE_Message( Admin_Exception(__CONTEXT__,KERNEL_INIT_LOG_FAILED,
                                         "BackupLog in admin mode",
                                          Log_ErrorStateStrings[result]) );
            Offline ();
        }
    }
    
    if ( completeLog )
    {
        k38init_save ( backupDescription,
                       m_State.IsAdmin(),
                       completeLog,
                       firstTimeTapeCheck );
        if ( e_ok == trans.trError_gg00 )
            k38headmaster (backupDescription);
            
        if ( m_State.IsAdmin() )
            Log_Volume::Instance().CloseLogArea(trans.trTaskId_gg00);

        m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
        return;
    }

    // backup each log segment separately

    bool                        anylogWasSaved = false;
    tgg00_MessBufHeader         lastResponse_mb_head;
    tgg00_SaveRestoreInputParam lastResponse_user_data;
    do
    {
        k38init_save ( backupDescription,
                       m_State.IsAdmin(),
                       completeLog,
                       firstTimeTapeCheck );
        if ( e_ok != trans.trError_gg00 )
            break;
        if ( e_ok != trans.trError_gg00 )
            break;
        k38headmaster (backupDescription);                
        if ( e_ok != trans.trError_gg00 )
            break;
        anylogWasSaved = true;
        
        lastResponse_mb_head   = backupDescription.mb_header();
        lastResponse_user_data = backupDescription.mb_qual()->msave_restore();
        
        backupDescription.mb_header()                = initial_mb_head;
        backupDescription.mb_qual()->msave_restore() = inital_user_data;
        backupDescription.mb_qual_len()              = sizeof (inital_user_data);
    }
    while ( ! Log_Volume::Instance().GetLogSaveIter().IsLastSegmentToSave() );

    if ( e_new_hostfile_required == trans.trError_gg00 )
    {
        // the current log backup is cancelled now
        backupDescription.mb_header()        = initial_mb_head;
        backupDescription.mb_qual_len()      = sizeof (inital_user_data);
        backupDescription.mb_type().becomes  (m_save_parallel);
        backupDescription.mb_type2().becomes (mm_abort);
        trans.trError_gg00                   = e_ok;
        k38headmaster (backupDescription);
        trans.trError_gg00 = e_cancelled;
        return; // do not detach devices
    }

    if ( e_ok == trans.trError_gg00
         ||
         e_no_log_to_save == trans.trError_gg00 )
    {
        if ( anylogWasSaved )
        {
            trans.trError_gg00                           = e_ok;
            backupDescription.mb_header()                = lastResponse_mb_head;
            backupDescription.mb_qual_len()              = sizeof (lastResponse_user_data);
            backupDescription.mb_qual()->msave_restore() = lastResponse_user_data;
        }
        else
        {
            backupDescription.mb_data_len() = 0;
            backupDescription.mb_struct().becomes(mbs_buf);
            backupDescription.mb_type().becomes  (m_return_result);
        }
    }

    if ( m_State.IsAdmin() )
        Log_Volume::Instance().CloseLogArea(trans.trTaskId_gg00);

    m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
}
/* --------------------------------------------------------------------------- */
01103 void Kernel_Administration::RestoreData (tgg00_TransContext &trans,
                                         tgg00_MessBlock    &backupDescription)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RestoreData", Common_Trace, 5);

    if ( m_State.IsOnline() )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Data Recovery","Online"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    if ( ! m_State.EnterCriticalState(Kernel_State::criticalStateBackup) )
    {
        RTE_Message (Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Backup"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    k38init_restore (backupDescription, IsCheckBackupConfigured());
    if ( e_ok == trans.trError_gg00 )
        k38headmaster (backupDescription);

    if (    e_new_hostfile_required != trans.trError_gg00 
         && e_wrong_hostfile        != trans.trError_gg00 
         && e_invalid_label         != trans.trError_gg00 )
    {
        m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
    }
}
/* --------------------------------------------------------------------------- */
01133 void Kernel_Administration::BackupRestoreHandling (tgg00_TransContext &trans,
                                                   tgg00_MessBlock    &handlingDescription)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::BackupRestoreHandling", Common_Trace, 5);

    if ( handlingDescription.mb_type2() == mm_fread )
    {
        // only for reading label of backup medium
        if ( ! m_State.EnterCriticalState(Kernel_State::criticalStateBackup) )
        {
            trans.trError_gg00 = e_dbm_command_not_possible;
            return;
        }
        k38headmaster (handlingDescription);
        m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
        return;
    }
    
    if ( ! m_State.IsCriticalState(Kernel_State::criticalStateBackup) )
      {
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
      }
    
    k38headmaster (handlingDescription);
    
    if ( trans.trError_gg00 == e_ok )
    {
        m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
        if ( m_State.IsCriticalState(Kernel_State::criticalStateRedo) )
            m_State.LeaveCriticalState(Kernel_State::criticalStateRedo);
    }
    else
    {
        if ( handlingDescription.mb_type2() == mm_abort )                  // PTS 1133694 UH 2005-02-01 added leaveCriticalState
            m_State.LeaveCriticalState(Kernel_State::criticalStateBackup);
            
        if ( m_State.IsCriticalState(Kernel_State::criticalStateRedo) )
        // in case of an error during recovery/redo the database has to be stopped
        {
            if (    trans.trError_gg00 != e_new_hostfile_required 
                 && trans.trError_gg00 != e_wrong_hostfile
                 && trans.trError_gg00 != e_hostfile_error // PTS 1124787 mb 2003-10-23
                 && trans.trError_gg00 != e_incompatible_log 
                 && trans.trError_gg00 != e_invalid_label 
                 && trans.trError_gg00 != e_save_skipped )
            {
                Offline (trans.trError_gg00);
            }
        }
    }
}
/* --------------------------------------------------------------------------- */
01186 void Kernel_Administration::InitConfig (tgg00_TransContext &trans)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::InitConfig", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    if ( m_State.IsOnline() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Reinstallation of database","Online"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    if ( ! state.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    
    if ( IsHotStandbyConfigured() )
        ForceHotStandbyRoleToMaster();

    tsp00_Line  DBIdent;

    vget_uniqe_id (DBIdent);

    b01create_filesystem (trans);
    if ( trans.trError_gg00 != e_ok )
        Offline (trans.trError_gg00);

    b01fcreate_file (trans, g01tabid.sys1_cat, 0, false);
    if ( trans.trError_gg00 != e_ok )
        Offline (trans.trError_gg00);

    b01fcreate_file (trans, g01tabid.sys2_cat, 0, false);
    if ( trans.trError_gg00 != e_ok )
        Offline (trans.trError_gg00);

    k57create_restartrecord (trans);
    if ( trans.trError_gg00 != e_ok )
        Offline (trans.trError_gg00);

    bd20FlushDataCache (trans);
    if ( trans.trError_gg00 != e_ok )
        Offline (trans.trError_gg00);

    Converter_Version           converterVersion;
    Data_PageNo                 maxStaticPageNo;
    Data_PageNo                 maxDynamicPageNo;
    IOMan_PackedBlockAddress    packedRootBlockAddr;

    if( ! Converter_IManager::Instance().FlushSerial (trans.trTaskId_gg00, converterVersion, 
            maxStaticPageNo, maxDynamicPageNo, packedRootBlockAddr))
    {
        Offline (e_shutdown);
    }

    k57restartrec->rstConfigParam_kb00().crFdirRoot_kb00          = bd17GetFdirRoot();
    k57restartrec->rstConfigParam_kb00().crLongDirRoot_kb00       = bd17GetLongFdirRoot();
    k57restartrec->rstConfigParam_kb00().crMaxDynamicPno_kb00     = maxDynamicPageNo;
    k57restartrec->rstConfigParam_kb00().crMaxStaticPno_kb00      = maxStaticPageNo;
    k57restartrec->rstConfigParam_kb00().crConvRootBlockAddr_kb00 = packedRootBlockAddr;
    k57restartrec->rstConverterVersion_kb00()                     = converterVersion;

    memcpy (k57restartrec->rstDbIdent_kb00(), DBIdent, sizeof(DBIdent));

    Log_Volume &log = Log_Volume::Instance();
    const Log_ErrorState result = log.CreateForNewDatabase (trans.trTaskId_gg00);

    if ( result != Log_Ok )
    {
        log.WriteDeviceParametersToTrace();
        RTE_Message( Admin_Exception(__CONTEXT__,KERNEL_INIT_LOG_FAILED,
                     "CreateForNewDatabase", Log_ErrorStateStrings[result]) );
        Offline (true);
    }
    
    k57restartrec->rstConfigPhase_kb00().becomes(cfpConfigured_ekb00);
    FlushRestartRecord (trans.trTaskId_gg00);
}
/* --------------------------------------------------------------------------- */
01268 void Kernel_Administration::AddLogVolume (tgg00_TransContext &trans,
                                          tsp00_VFilename    &volumeName,
                                          tsp00_VFilename    &mirrorVolumeName,
                                          SAPDB_Int4          newVolumeSize,
                                          SAPDB_Int2          volumeId)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::AddLogVolume", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    if ( ! state.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    if ( ! memcmp (volumeName, b01blankfilename, sizeof(b01blankfilename))
         ||
         (IOMan_ILogInfo::GetInstance().IsLogMirrored()
          &&
          ! memcmp (mirrorVolumeName, b01blankfilename, sizeof(b01blankfilename))) )
      {
        trans.trError_gg00 = e_devname_invalid;
        return;
      }
    
    if ( newVolumeSize <= 0 )
      {
        trans.trError_gg00 = e_devsize_invalid;
        return;
      }

    FrameCtrl_ILog          &frameControl = FrameCtrl_ILog::GetInstance();
    Kernel_IPage::PageFrame  pageFrame = frameControl.NewLogFrame(trans.trTaskId_gg00);
    Log_Page                 page  (pageFrame, Log_QueueID(), Log_IOSequenceNo());
    IOMan_LogPages           pages (*(reinterpret_cast<SAPDBMem_IRawAllocator*>(trans.trAllocator_gg00)),UTF8("AddLogVolume"));
    IOMan_BlockCount         newLogSize = 0;

    if ( ! pages.Initialize(g01io_block_count()) )
    {
        frameControl.FreeLogFrame (trans.trTaskId_gg00, pageFrame);
        trans.trError_gg00 = e_sysbuf_storage_exceeded;
        return;
    }

    for( SAPDB_Int i = 0; i < g01io_block_count(); ++i )
        pages.Add( page );

    if ( ! IOMan_ILogManager::GetInstance().AddLogVolume
                               ( trans.trTaskId_gg00,
                                 m_State.IsAdmin(),
                                 volumeName,
                                 mirrorVolumeName,
                                 newVolumeSize,
                                 volumeId,
                                 pages,
                                 newLogSize ) )
    {
        frameControl.FreeLogFrame (trans.trTaskId_gg00, pageFrame);
        trans.trError_gg00 = e_new_disk_not_accessible;
        return;
    }                                

    frameControl.FreeLogFrame (trans.trTaskId_gg00, pageFrame);
    if ( newLogSize <= 0 )
    {
        Kernel_VTrace() << "AddLogVolume: Invalid new logsize: " << newLogSize;
        RTE_Crash( SAPDBErr_Exception(__CONTEXT__,SAPDBERR_ASSERT_STATE_FAILED,
                                      "AddLogVolume: Invalid new logsize ") );
    }
    Log_Volume::Instance().SetNewDeviceEndOffset(newLogSize - 1);
}
/* --------------------------------------------------------------------------- */
01342 void Kernel_Administration::AddDataVolume (tgg00_TransContext &trans,
                                           tsp00_VFilename     &volumeName,
                                           SAPDB_Int4           newVolumeSize,
                                           SAPDB_Int2           volumeId)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::AddDataVolume", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    if ( ! state.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    if ( ! memcmp (volumeName, b01blankfilename, sizeof(b01blankfilename)) )
      {
        trans.trError_gg00 = e_devname_invalid;
        return;
      }
    
    if ( newVolumeSize <= 0 )
      {
        trans.trError_gg00 = e_devsize_invalid;
        return;
      }

    if ( ! IOMan_IDataManager::GetInstance().AddDataVolume( trans.trTaskId_gg00,
                                                            m_State.IsAdmin(),
                                                            volumeName,
                                                            newVolumeSize,
                                                            volumeId ) )
    {
        trans.trError_gg00 = e_new_disk_not_accessible;
        return;
    }
}
/* --------------------------------------------------------------------------- */
01381 void Kernel_Administration::GetRedoLogProgressInfo (SAPDB_UInt &transread,
                                                    SAPDB_UInt &transredone)
{
    m_RedoManager.GetProgressInfo(transread,transredone);
}
/* --------------------------------------------------------------------------- */
01387 void Kernel_Administration::GetRestartVersion (tgg00_TransContext &trans,
                                               Log_IOSequenceNo   &firstUsedIOsequence,
                                               Log_IOSequenceNo   &firstLogIOsequence,
                                               SAPDB_Char         *dbidentRestartrecord,
                                               SAPDB_UInt          dbidentRestartrecordSize,
                                               SAPDB_Char         *dbidentLogInfo,
                                               SAPDB_UInt          dbidentLogInfoSize,
                                               RTE_Nodename       &masterNodeName,
                                               bool               &logWritingEnabled,
                                               bool               &logAutoOverwrite,
                                               bool               &restartable,
                                               SAPDB_Int4         &converterVersion)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::GetRestartVersion", Common_Trace, 5);

    if ( dbidentRestartrecordSize != dbidentLogInfoSize )
    {
        trans.trError_gg00 = e_not_implemented;
        return;
    }
    
    tsp00_TaskId     &taskid  = trans.trTaskId_gg00;
    tgg00_BasisError &trError = trans.trError_gg00;

    Kernel_StateScope state1 (Kernel_State::criticalStateConfiguration, m_State);
    Kernel_StateScope state2 (Kernel_State::criticalStateBackup, m_State);

    const bool volumesAreAttached  = ! m_State.IsAdmin() || m_State.IsRedo();
    
    if ( ! volumesAreAttached )
    {
        if ( ! state1.EnterCriticalState() )
        {
            RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
            trans.trError_gg00 = e_dbm_command_not_possible;
            return;
        }

        if ( ! state2.EnterCriticalState() )
        {
            RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Backup"));
            trans.trError_gg00 = e_dbm_command_not_possible;
            return;
        }

        if ( 0 == k57restartrec )
        {
            trError = e_no_more_memory;
            return;
        }

        if ( ! ReadRestartRecord (taskid) )
        {
            trError = e_disk_not_accessible;
            return;
        }

        state2.LeaveCriticalState();
        state1.LeaveCriticalState();
    }

    // determine first needed iosequence to restart and dbident of data area
    
          Log_Volume & log      = Log_Volume::Instance(); 
    const SAPDB_Char * pDbIdent = reinterpret_cast<const SAPDB_Char*>(&(k57restartrec->rstDbIdent_kb00()));
    
    converterVersion    = k57restartrec->rstConverterVersion_kb00();
    firstUsedIOsequence = k57restartrec->rstLastSavept_kb00().svpIOsequence_kb00;
    g10mv1 (__FILE__,  1,
          sizeof (tsp00_Line), dbidentRestartrecordSize,
          pDbIdent, 1,
          dbidentRestartrecord, 1,
          sizeof (tsp00_Line), trError);

    // determine all log parameters

    if ( ! volumesAreAttached )
    {
        if ( ! state1.EnterCriticalState() )
        {
            RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
            trans.trError_gg00 = e_dbm_command_not_possible;
            return;
        }

        if ( ! state2.EnterCriticalState() )
        {
            RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Backup"));
            trans.trError_gg00 = e_dbm_command_not_possible;
            return;
        }

        const Log_ErrorState  result = log.RestartForSaveLogCold(taskid);
    
        if ( result != Log_Ok )
        {
            log.WriteDeviceParametersToTrace();
            RTE_Message( Admin_Exception(__CONTEXT__,KERNEL_INIT_LOG_FAILED,
                         "GetRestartVersion", Log_ErrorStateStrings[result]) );
            log.CloseLogArea (taskid);
            trError = e_log_error;
            return;
        }
    }

    log.GetRestartVersion ( taskid,
                            firstLogIOsequence,
                            dbidentLogInfo, dbidentLogInfoSize,
                            masterNodeName );
    logWritingEnabled = log.IsDevspaceEnabled(taskid);
      logAutoOverwrite  = log.IsAutomaticOverwrite(taskid);
    if (log.LogIsEmpty()) //rewritten with 1124724 mb 2003-11-13
    {
        restartable = true;
    }
    else
    {
        restartable = firstUsedIOsequence.IsInRange ( firstLogIOsequence,log.GetLastKnownIOSeq());
        // check also the db-identifier
        restartable = restartable && ! memcmp(dbidentRestartrecord, dbidentLogInfo, dbidentRestartrecordSize);
    }

    if ( ! volumesAreAttached )
        log.CloseLogArea(taskid);
}
/* --------------------------------------------------------------------------- */
01513 bool Kernel_Administration::ReadRestartRecord (tsp00_TaskId taskid)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::ReadRestartRecord", Common_Trace, 5);

    if ( 0 == k57restartrec )
        return false;

    const IOMan_BlockAddress    restartInfoAddr = IOMan_ReservedBlockAddress().GetRestartPageAddress();
    const IOMan_DataVolumeInfo  restartInfoVolume ( restartInfoAddr.GetDeviceNo() );
          Kernel_RestartPage    restartInfoPage   (k57restartrec);
        
    if ( restartInfoVolume.IsVolumeOnline() )
        IOMan_IDataIO::GetInstance().ReadRestartPage(taskid, restartInfoPage);

    IOMan_IDataManager &ioManData = IOMan_IDataManager::GetInstance();

    if ( ! ioManData.OpenOneDataVolume(taskid, restartInfoAddr.GetDeviceNo()) )
        return false;

    IOMan_IDataIO::GetInstance().ReadRestartPage(taskid, restartInfoPage);

    ioManData.CloseAllDataVolumes(taskid);

    return true;
}
/* --------------------------------------------------------------------------- */
01539 bool Kernel_Administration::FlushRestartRecord (tsp00_TaskId taskid)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::FlushRestartRecord", Common_Trace, 5);

    const IOMan_BlockAddress    restartInfoAddr = IOMan_ReservedBlockAddress().GetRestartPageAddress();
    const IOMan_DataVolumeInfo  restartInfoVolume ( restartInfoAddr.GetDeviceNo() );

    if ( restartInfoVolume.IsVolumeOnline() )
    {
        k57save_restartrecord(taskid);
        return true;
    }

    IOMan_IDataManager &ioManData = IOMan_IDataManager::GetInstance();
    
    if ( ! ioManData.OpenOneDataVolume(taskid, restartInfoAddr.GetDeviceNo()) )
        return false;

    k57save_restartrecord(taskid);
    
    ioManData.CloseAllDataVolumes(taskid);
    return true;
}
/* --------------------------------------------------------------------------- */
01563 bool Kernel_Administration::IsCheckBackupConfigured()
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::IsCheckBackupConfigured", Common_Trace, 5);

    tsp00_VFilename           devicename;
    const IOMan_BlockAddress  restartRecordBlockAddress = IOMan_ReservedBlockAddress().GetRestartPageAddress();

    IOMan_IDataInfo::GetInstance().GetDataVolumeName( restartRecordBlockAddress.GetDeviceNo(),
                                                   devicename );
    return ! memcmp (devicename,"CHECKDEV",8);
}
/* --------------------------------------------------------------------------- */
01575 void Kernel_Administration::GetNewCommandId( tsp00_TaskId     taskid,
                                             tgg00_UtilCmdId &id )
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::GetNewCommandId", Common_Trace, 5);

    SAPDB_UInt2 commandno;
    SAPDB_Int4  timestamp;

    vtime_in_seconds (timestamp);

    if ( g01code.kernel_swap != sw_normal )
        s20ch4sw (timestamp, g01code.kernel_swap, &timestamp, 1, sw_normal);

    // ************** SYNCH ******************
    m_SpinLock.Lock();
    m_CommandNo = (m_CommandNo+1) % 999;
    commandno   = m_CommandNo;
    m_SpinLock.Unlock();
    // ************** END SYNCH **************

    SAPDB_Byte* auxint = reinterpret_cast<SAPDB_Byte*>(&timestamp);
    SAPDB_Char* auxstr = reinterpret_cast<SAPDB_Char*>(&id);

    SAPDB_sprintf(auxstr, sizeof(id.utidId_gg00), "%02X%02X%02X%02X", auxint[0], auxint[1], auxint[2], auxint[3]);

    auxstr += 8;
    auxint  = reinterpret_cast<SAPDB_Byte*>(&commandno);
    
    if ( g01code.kernel_swap == sw_normal )
        SAPDB_sprintf(auxstr, sizeof(id.utidId_gg00)-7, "%02X%02X", auxint[0], auxint[1]);
    else
        SAPDB_sprintf(auxstr, sizeof(id.utidId_gg00)-7, "%02X%02X", auxint[1], auxint[0]);
    id .utidLineNo_gg00 = 0;
} 
/* --------------------------------------------------------------------------- */
01610 void Kernel_Administration::ReIntegrateBadLogVolume (tgg00_TransContext &trans,
                                                     tsp00_VFilename    &devicename)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::ReIntegrateBadLogVolume", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    if ( m_State.IsOnline() ) // PTS 1123452 UH 2003-08-06 removed "!"
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Reintegrate log volume","Online"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    if ( ! state.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    if( ! IOMan_ILogManager::GetInstance().ReIntegrateBadLogVolume(trans.trTaskId_gg00,devicename) )
        trans.trError_gg00 = e_disk_not_accessible;
}
/* --------------------------------------------------------------------------- */
01634 void Kernel_Administration::GetBadLogVolume (tgg00_TransContext &trans,
                                             tsp00_VFilename    &devicename)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::GetBadLogVolume", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    if ( m_State.IsOnline() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Get bad log volume","Online"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    if ( ! state.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    
    if( ! IOMan_ILogInfo::GetInstance().GetBadLogVolume(trans.trTaskId_gg00,devicename) )
        trans.trError_gg00 = e_row_not_found; // no bad volume found
}
/* --------------------------------------------------------------------------- */
01658 void Kernel_Administration::SetLogWriting (tsp00_TaskId taskid,
                                           bool         on)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::SetLogWriting", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    if ( ! m_State.IsAdmin() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"SetLogWriting","not Admin"));
        Offline (e_dbm_command_not_possible, true);
    }
    if ( ! state.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        Offline (e_dbm_command_not_possible, true);
    }

    if ( ! Log_Volume::Instance().SetLogWriting (taskid, on) )
        Offline (e_disk_not_accessible, true);
}
/* --------------------------------------------------------------------------- */
01680 bool Kernel_Administration::IsLogWritingEnabled (tsp00_TaskId taskid)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::IsLogWritingEnabled", Common_Trace, 5);
    return Log_Volume::Instance().IsDevspaceEnabled(taskid);
}
/* --------------------------------------------------------------------------- */
01686 void Kernel_Administration::SetLogAutoOverwrite (tsp00_TaskId taskid,
                                                 bool         on)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::SetLogAutoOverwrite", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    if ( ! state.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        Offline (e_dbm_command_not_possible, true);
    }

    if ( ! Log_Volume::Instance().SetAutomaticOverwrite (taskid, on) )
        Offline (e_disk_not_accessible, true);
}
/* --------------------------------------------------------------------------- */
01703 bool Kernel_Administration::IsLogAutoOverwriteEnabled (tsp00_TaskId taskid)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::IsLogAutoOverwriteEnabled", Common_Trace, 5);
    return Log_Volume::Instance().IsAutomaticOverwrite(taskid);
}
/* --------------------------------------------------------------------------- */
01709 bool Kernel_Administration::KernelStateIsRestart()
{
    return m_State.IsRedo();
}
/* --------------------------------------------------------------------------- */
01714 bool Kernel_Administration::KernelStateIsOnline()
{
    return m_State.IsOnline();
}
/* --------------------------------------------------------------------------- */
01719 bool Kernel_Administration::KernelStateIsAdmin()
{
    return m_State.IsAdmin();
}
/* --------------------------------------------------------------------------- */
01724 void Kernel_Administration::GetBlockAddress ( 
                                             const tsp00_TaskId          taskid,
                                             const Data_PageNo           &pageno,
                                             const Data_PageRecoveryMode &recMode,
                                             IOMan_BlockAddress          &block )
{
    SAPDBTRACE_METHOD_DEBUG( "Kernel_Administration::GetBlockAddress", Common_Trace, 5 );
    
    Kernel_StateScope state( Kernel_State::criticalStateConfiguration, m_State );
    
    if( m_State.IsAdmin() && ! state.EnterCriticalState() )
    {
        RTE_Message( Admin_Exception( __CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration" ));
        block = IOMan_BlockAddress();
        return;
    }
    
    tgg00_BasisError error = e_ok;
    block.Invalidate();
    
    if( m_State.IsAdmin() )
        bd14RestartFileSystemForSaveDataCold (taskid, error);
    
    if( e_ok == error )
        Converter_IDiagnose::Instance().GetBlockAddress (taskid, pageno, recMode, block);
    
    if( m_State.IsAdmin() )
        bd14ShutdownFileSystemAfterSaveDataCold (taskid);
}
/* --------------------------------------------------------------------------- */
01754 bool Kernel_Administration::SetNewBlockAddress ( 
                                                const tsp00_TaskId          taskid,
                                                const Data_PageNo           &pageno,
                                                const Data_PageRecoveryMode &recMode,
                                                const IOMan_BlockAddress    &block )
{
    SAPDBTRACE_METHOD_DEBUG( "Kernel_Administration::SetNewBlockAddress", Common_Trace, 5 );

    Kernel_StateScope state( Kernel_State::criticalStateConfiguration, m_State );

    if ( m_State.IsAdmin() && ! state.EnterCriticalState() )
    {
        RTE_Message( Admin_Exception( __CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration" ));
        return false;
    }

    tgg00_BasisError error = e_ok;
    bool             result;

    if( m_State.IsAdmin() )
        bd14RestartFileSystemForSaveDataCold (taskid, error);

    if( e_ok == error )
        result = Converter_IDiagnose::Instance().SetNewBlockAddress( taskid, pageno, recMode, block );
      else
        result = false;

    if ( m_State.IsAdmin() )
    {
        if( result )
        {
            IOMan_PackedBlockAddress  packedRootBlockAddr;

            if( Converter_IManager::Instance().FlushSerialForColdSave( taskid, packedRootBlockAddr ))
            {
                k57restartrec->rstConfigParam_kb00().crConvRootBlockAddr_kb00 = packedRootBlockAddr;

                FlushRestartRecord (taskid);
            }
        }
        bd14ShutdownFileSystemAfterSaveDataCold (taskid);
    }

    return result;
}
/* --------------------------------------------------------------------------- */
01800 bool Kernel_Administration::DeleteBlockAddress( 
                                               const tsp00_TaskId          taskid,
                                               const Data_PageNo           &pageno,
                                               const Data_PageRecoveryMode &recMode )
{
    SAPDBTRACE_METHOD_DEBUG( "Kernel_Administration::DeleteBlockAddress", Common_Trace, 5 );

    Kernel_StateScope state( Kernel_State::criticalStateConfiguration, m_State );

    if( m_State.IsAdmin() && ! state.EnterCriticalState() )
    {
        RTE_Message( Admin_Exception( __CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration" ));
        return false;
    }

    tgg00_BasisError error = e_ok;
    bool             result;

    if( m_State.IsAdmin() )
        bd14RestartFileSystemForSaveDataCold (taskid, error);

    if( e_ok == error )
        result = Converter_IDiagnose::Instance().DeleteBlockAddress( taskid, pageno, recMode );
      else
        result = false;

    if( m_State.IsAdmin() )
    {
        if( result )
        {
            IOMan_PackedBlockAddress  packedRootBlockAddr;

            if (Converter_IManager::Instance().FlushSerialForColdSave( taskid, packedRootBlockAddr ))
            {
                k57restartrec->rstConfigParam_kb00().crConvRootBlockAddr_kb00 = packedRootBlockAddr;

                FlushRestartRecord (taskid);
            }
        }
        bd14ShutdownFileSystemAfterSaveDataCold (taskid);
    }

    return result;
}
/* --------------------------------------------------------------------------- */
01845 bool Kernel_Administration::SuspendLogWriter ( tsp00_TaskId      taskid,
                                               Log_IOSequenceNo &lastWrittenIOsequence)
{
    // PTS 1114053 UH 2002-10-22 new as kb560SuspendAndGetLastWrittenIOSequence
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::SuspendLogWriter", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    lastWrittenIOsequence = Log_IOSequenceNo();
    
    if ( m_State.IsAdmin()  )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"SuspendLogWriter","Admin"));
        return false;
    }
    if ( ! Log_Volume::IsReadyForWriting()  )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"SuspendLogWriter","log volume is not initialized"));
        return false;
    }
    if ( ! Log_SavepointManager.DenySavepoint (taskid) )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"SuspendLogWriter","DenySavepoint failed"));
        return false;
    }
    if ( ! state.EnterCriticalState() ) // PTS 1121281 UH 2003-03-24
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        return false;
    }
    return Log_Volume::Instance().SuspendAndGetLastWrittenIOSequence(taskid, lastWrittenIOsequence);
}
/* --------------------------------------------------------------------------- */
01878 void Kernel_Administration::ResumeLogWriter (tsp00_TaskId taskid)
{
    // PTS 1114053 UH 2002-10-22 new as k560ResumeLogWriterByUser
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::ResumeLogWriter", Common_Trace, 5);

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    if ( m_State.IsAdmin() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"ResumeLogWriter","Admin"));
        return;
    }
    if ( ! Log_Volume::IsReadyForWriting() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"ResumeLogWriter","Log volume not initialized"));
        return;
    }
    if ( ! state.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        return;
    }
      Log_SavepointManager.PermitSavepoint (taskid); // PTS 1121281 UH 2003-03-24
    Log_Volume::Instance().ResumeByUser();
}

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

01906 void Kernel_Administration::CheckData( 
                                      tgg00_TransContext  &trans,
                                      const SAPDB_Bool    bWithExtendedCheck,
                                      const SAPDB_Bool    bWithIndexes,
                                      SAPDB_Int4          &badIndexCount,
                                      SAPDB_Int4          &totalBadCount )
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::CheckData", Common_Trace, 5);
    
    Kernel_StateScope state( Kernel_State::criticalStateCheckData, m_State );
    
    if( ! ( m_State.IsOnline() || m_State.IsRedo() )) // PTS 1127504 2004-02-03
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"CheckData","Redo"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    if ( ! state.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"CheckData"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    SAPDB_Int4       pagesUsed      = 0;
    SAPDB_Int4       blocksReleased = 0;
    const SAPDB_Bool bWithUpdate    = false;

    b01verify_database( trans, bWithUpdate, bWithIndexes, bWithExtendedCheck,
        pagesUsed, blocksReleased, badIndexCount, totalBadCount );
}

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

01941 void Kernel_Administration::CheckDataWithUpdate( 
                                                tgg00_TransContext  &trans,
                                                const SAPDB_Bool    bWithExtendedCheck,
                                                SAPDB_Int4          &pagesUsed,
                                                SAPDB_Int4          &blocksReleased,
                                                SAPDB_Int4          &badIndexCount,
                                                SAPDB_Int4          &totalBadCount )
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::CheckDataWithUpdate", Common_Trace, 5);
    
    Kernel_StateScope state1( Kernel_State::criticalStateConfiguration, m_State );
    Kernel_StateScope state2( Kernel_State::criticalStateBackup, m_State );
    Kernel_StateScope state3( Kernel_State::criticalStateCheckData, m_State );
    Kernel_StateScope state4( Kernel_State::criticalStateRedo, m_State ); 
   

    // Note that "check data with update" and "save data/pages" are using the same
    // backup pno list within the converter, therefore no concurrent access is
    // possible. Configuration latch is used to hold the data area respl. converter
    // entries stable.
    // Redo state is set to guarantee that no restart is possible as long as check
    // data with update is executed and therefore no filesystem restart is made
    
    if( ! m_State.IsAdmin() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"CheckDataWithUpdate","not Admin"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    if( ! state1.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    if( ! state2.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Backup"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    if( ! state3.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"CheckData"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }
    if( ! state4.EnterCriticalState() ) // PTS 1126032 TS 2003-08-12
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Redo"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    // Do not increment converter version, because no savepoint will be generated!
    
    if( ! RestartFilesystem(trans, false) )
        Offline(trans.trError_gg00);
    
    const SAPDB_Bool bWithIndexes = true; // Check all indexes
    const SAPDB_Bool bWithUpdate  = true; // Free unreferenced blocks
    
    b01verify_database( trans, bWithUpdate, bWithIndexes, bWithExtendedCheck,
        pagesUsed, blocksReleased, badIndexCount, totalBadCount );
    
    const tgg00_BasisError  auxError = trans.trError_gg00;
    b01shutdown_filesystem( trans );
    trans.trError_gg00 = auxError;
}

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

02013 void Kernel_Administration::Diagnose( tgg00_MessBlock &messBlock )
{
    SAPDBTRACE_METHOD_DEBUG( "Kernel_Administration::Diagnose", Common_Trace, 5 );


    const SAPDB_Int4        pageSize   = g01page_size;
    const tgg00_DiagType    &diagType  = messBlock.mb_qual()->mut_diag_type();
    const bool              bIsRestore = ( mm_device_restore == messBlock.mb_type2());

    tgg00_TransContext      &trans   = *(messBlock.mb_trns());
    tgg00_BasisError        &trError = trans.trError_gg00;
    
    // Check preconditions for diagnose repair

    Kernel_StateScope state1( Kernel_State::criticalStateConfiguration, m_State );
    Kernel_StateScope state2( Kernel_State::criticalStateBackup, m_State );
    Kernel_StateScope state3( Kernel_State::criticalStateCheckData, m_State );
    Kernel_StateScope state4( Kernel_State::criticalStateRedo, m_State ); 

    if( bIsRestore )
    {
        if(  ! m_State.IsAdmin())
        {
            RTE_Message( Admin_Exception( __CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,
                "Diagnose repair", "state is not Admin" ));
            trError = e_dbm_command_not_possible;
            return;
        }
        if( ! state1.EnterCriticalState())
        {
            RTE_Message( Admin_Exception( __CONTEXT__, KERNEL_CRITICAL_STATE_NOT_ENTERED, "Configuration" ));
            trError = e_dbm_command_not_possible;
            return;
        }
        if( ! state2.EnterCriticalState())
        {
            RTE_Message( Admin_Exception( __CONTEXT__, KERNEL_CRITICAL_STATE_NOT_ENTERED, "Backup" ));
            trError = e_dbm_command_not_possible;
            return;
        }
        if( ! state3.EnterCriticalState())
        {
            RTE_Message( Admin_Exception( __CONTEXT__, KERNEL_CRITICAL_STATE_NOT_ENTERED, "CheckData" ));
            trError = e_dbm_command_not_possible;
            return;
        }
        if( ! state4.EnterCriticalState())
        {
            RTE_Message( Admin_Exception( __CONTEXT__, KERNEL_CRITICAL_STATE_NOT_ENTERED, "Redo" ));
            trError = e_dbm_command_not_possible;
            return;
        }
    }

    if( diagNil_egg00 == diagType ){
        trError = e_not_implemented;
        return;
    }
    if( messBlock.mb_data_size() < pageSize ){
        trError = e_too_short_datapart;
        return;
    }
    if(( bIsRestore ) && ( pageSize != messBlock.mb_data_len()))
    {
        trError = e_too_short_datapart;
        return;
    }
    tsp00_PageAddr  pPage = reinterpret_cast< tsp00_PageAddr >
        ( RTE_ISystem::Instance().AllocSystemPages( pageSize ));
    
    if( NULL == pPage ){
        trError = e_no_more_memory;
        return;
    }
    if( bIsRestore )
    {  // copy page to be restored into i/o aligned buffer
        memcpy( pPage, messBlock.mb_data()->mbp_buf(), pageSize );
    }

    bool bPageWasExtracted    = false;
    bool bShutdownFileSystem  = false;
    bool bCloseVolumes        = false;

    if(
        ( diagDev_egg00         != diagType ) &&
        ( diagLoginfoPage_egg00 != diagType ) &&
        ( diagLogDev_egg00      != diagType ) &&
        ( diagMirrLogDev_egg00  != diagType ) &&
        ( diagRestart_egg00     != diagType ) &&
        ( b01downfilesystem ) // +++ get file system state
        )
    {
        RestartFilesystem( trans, false ); // do not increment converter version
        bShutdownFileSystem = true;
    }
    
    if( e_ok == trError )
    {
        if( ! DiagnoseHandling( trans, messBlock, pPage, 
            diagType, bIsRestore, bPageWasExtracted, bCloseVolumes ))
        {
            if( e_ok == trError )
                trError = e_disk_not_accessible;
        }
    }

    if( NULL != pPage ){
        RTE_ISystem::Instance().FreeSystemPages( pPage, pageSize );
    }

    if( bShutdownFileSystem )
    {
        const tgg00_BasisError  auxError = trans.trError_gg00;
        b01shutdown_filesystem( trans );
        trans.trError_gg00 = auxError;
    }
    else if( bCloseVolumes )
    {
        IOMan_IDataManager::GetInstance().CloseAllDataVolumes( trans.trTaskId_gg00 );
        Log_Volume::Instance().CloseLogArea( trans.trTaskId_gg00 );
    }

    if(( e_ok == trError ) && bPageWasExtracted )
    {
        messBlock.mb_type().becomes( m_return_result );
        messBlock.mb_type2().becomes( mm_nil );
        messBlock.mb_struct().becomes( mbs_buf );
    }
}

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

02145 bool Kernel_Administration::DiagnoseHandling(
                                             tgg00_TransContext     &trans, 
                                             tgg00_MessBlock        &messBlock,
                                             tsp00_PageAddr         pPage,
                                             const tgg00_DiagType   diagType,
                                             const bool             bIsRestore,
                                             bool                   &bPageWasExtracted,
                                             bool                   &bCloseVolumes )
{
    const tsp00_TaskId  &taskId         = trans.trTaskId_gg00;
    const SAPDB_Int4    pageSize        = g01page_size;
    const bool          bFromPrimaryLog = true;
    const bool          bToPrimaryLog   = true;

    tgg00_BasisError    &trError = trans.trError_gg00;


    switch( diagType ) 
    {
    case diagDev_egg00:
        {
            bool                bIsPrimaryLog;
            IOMan_DeviceNo      volumeNo; // determine with volume name
            const IOMan_BlockNo blockNo     = messBlock.mb_qual()->mut_pno();
            const IOMan_BlockNo stopBockNo  = messBlock.mb_qual()->mut_pno2();

            if( blockNo != stopBockNo )
            {
                trError = e_not_implemented;
                return false;
            }
            if( IOMan_IDataInfo::GetInstance().GetDataVolumeNo( // is data page !
                messBlock.mb_qual()->mut_dev(), volumeNo ))
            {
                const IOMan_BlockAddress    dataBlock( volumeNo, blockNo );
                const IOMan_DataVolumeInfo  dataVolumeInfo( dataBlock.GetDeviceNo());
                
                bCloseVolumes = ! dataVolumeInfo.IsVolumeOnline();

                if( ! IOMan_IDataManager::GetInstance().OpenOneDataVolume( 
                    taskId, dataBlock.GetDeviceNo())){
                    return false;
                }
                if( bIsRestore )
                {
                    return( IOMan_IDiagnose::GetInstance().WritePageDirectToDataVolume( 
                        taskId, dataBlock, pPage ));
                }
                // extract data page
                if( ! IOMan_IDiagnose::GetInstance().ReadPageDirectFromDataVolume( 
                    taskId, dataBlock, pPage )){
                    return false;
                } // else copy data page into order packet
            }
            else if( IOMan_ILogInfo::GetInstance().GetLogVolumeNo(  // is log page
                messBlock.mb_qual()->mut_dev(), volumeNo, bIsPrimaryLog ))
            {
                const IOMan_BlockAddress    logBlock( volumeNo, blockNo );
                const IOMan_LogVolumeInfo   logVolumeInfo( logBlock.GetDeviceNo());
                
                bCloseVolumes = ! logVolumeInfo.IsVolumeOnline();

                if( ! IOMan_ILogManager::GetInstance().OpenOneLogVolume( 
                    taskId, logBlock.GetDeviceNo())){
                    return false;
                }
                if( bIsRestore )
                {
                    if( bIsPrimaryLog ){
                        return( IOMan_IDiagnose::GetInstance().WritePageDirectToLogVolume(
                            taskId, logBlock, pPage, bToPrimaryLog ));
                    }
                    else{
                        return( IOMan_IDiagnose::GetInstance().WritePageDirectToLogVolume(
                            taskId, logBlock, pPage, ! bToPrimaryLog ));
                    }
                }
                // extract log page
                if( bIsPrimaryLog ){
                    if( ! IOMan_IDiagnose::GetInstance().ReadPageDirectFromLogVolume(
                        taskId, logBlock, pPage, bFromPrimaryLog )){
                        return false;
                    }
                }
                else{
                    if( ! IOMan_IDiagnose::GetInstance().ReadPageDirectFromLogVolume(
                        taskId, logBlock, pPage, ! bFromPrimaryLog )){
                        return false;
                    }
                } // else copy log page into order packet
            } 
            else // unknown volume
            {
                trError = e_devname_invalid;
                return false;
            }
            memcpy( messBlock.mb_data()->mbp_buf(), pPage, pageSize );
            messBlock.mb_data_len() = pageSize;
            bPageWasExtracted       = true;
            return true;
        };
    case diagLogDev_egg00:
    case diagMirrLogDev_egg00:
        {
            if( messBlock.mb_qual()->mut_pno() != messBlock.mb_qual()->mut_pno2() )
            {
                trError = e_not_implemented;
                return false;
            }
            const IOMan_BlockAddress    logBlock( messBlock.mb_qual()->mut_count(), messBlock.mb_qual()->mut_pno() );
            const IOMan_LogVolumeInfo   logVolumeInfo( logBlock.GetDeviceNo());
                
            bCloseVolumes = ! logVolumeInfo.IsVolumeOnline();

            if( ! IOMan_ILogManager::GetInstance().OpenOneLogVolume( 
                taskId, logBlock.GetDeviceNo())){
                return false;
            }
            if( bIsRestore )
            {
                if( diagLogDev_egg00 == messBlock.mb_qual()->mut_diag_type()){
                    return( IOMan_IDiagnose::GetInstance().WritePageDirectToLogVolume(
                        taskId, logBlock, pPage, bToPrimaryLog ));
                }
                else{
                    return( IOMan_IDiagnose::GetInstance().WritePageDirectToLogVolume(
                        taskId, logBlock, pPage, ! bToPrimaryLog ));
                }
            }
            // extract log page
            if( diagLogDev_egg00 == messBlock.mb_qual()->mut_diag_type())
            {
                if( ! IOMan_IDiagnose::GetInstance().ReadPageDirectFromLogVolume(
                    taskId, logBlock, pPage, bFromPrimaryLog )){
                    return false;
                }
            }
            else
            {
                if( ! IOMan_IDiagnose::GetInstance().ReadPageDirectFromLogVolume(
                    taskId, logBlock, pPage, ! bFromPrimaryLog )){
                    return false;
                }
            }
            memcpy( messBlock.mb_data()->mbp_buf(), pPage, pageSize );
            messBlock.mb_data_len() = pageSize;
            bPageWasExtracted       = true;
            return true;
        };
    case diagLoginfoPage_egg00:
        {
            const IOMan_BlockAddress    logInfoBlock( 1,1 ); // log info block
            const IOMan_LogVolumeInfo   logVolumeInfo( logInfoBlock.GetDeviceNo());
                
            bCloseVolumes = ! logVolumeInfo.IsVolumeOnline();

            if( ! IOMan_ILogManager::GetInstance().OpenOneLogVolume( 
                taskId, logInfoBlock.GetDeviceNo())){
                return false;
            }
            if( bIsRestore )
            {
                if( ! IOMan_IDiagnose::GetInstance().WritePageDirectToLogVolume(
                    taskId, logInfoBlock, pPage, bToPrimaryLog )){
                    return false;
                }
                if( ! IOMan_ILogInfo::GetInstance().IsLogMirrored()){
                    return true; // ready
                }
                if( ! IOMan_IDiagnose::GetInstance().WritePageDirectToLogVolume(
                    taskId, logInfoBlock, pPage, ! bToPrimaryLog )){
                    return false; 
                }
                return true; // ready
            }
            // extract log info page
            if( ! IOMan_IDiagnose::GetInstance().ReadPageDirectFromLogVolume(
                taskId, logInfoBlock, pPage, bFromPrimaryLog )){
                return false;
            }
            memcpy( messBlock.mb_data()->mbp_buf(), pPage, pageSize );
            messBlock.mb_data_len() = pageSize;
            bPageWasExtracted       = true;
            return true;
        };
    case diagRestart_egg00:
        {
            // get restart record block
            const IOMan_BlockAddress    rstBlock = 
                IOMan_ReservedBlockAddress().GetRestartPageAddress();
            const IOMan_DataVolumeInfo  dataVolumeInfo( rstBlock.GetDeviceNo());
            
            bCloseVolumes = ! dataVolumeInfo.IsVolumeOnline();
            
            if( ! IOMan_IDataManager::GetInstance().OpenOneDataVolume( 
                taskId, rstBlock.GetDeviceNo())){
                return false;
            }
            if( bIsRestore )
            {
                if( IOMan_IDiagnose::GetInstance().WritePageDirectToDataVolume( 
                    taskId, rstBlock, pPage )){
                    return true;
                } 
                return false;
            }
            // extract restart record 
            if( ! IOMan_IDiagnose::GetInstance().ReadPageDirectFromDataVolume( 
                taskId, rstBlock, pPage )){
                return false;
            }
            memcpy( messBlock.mb_data()->mbp_buf(), pPage, pageSize );
            messBlock.mb_data_len() = pageSize;
            bPageWasExtracted       = true;
            return true;
        };
    case diagPermPages_egg00:
    case diagStaticPages_egg00:
        {
            const bool                 bIsPermanent = ( diagPermPages_egg00 == diagType );
            const Data_AddressingMode  addrMode     = ( bIsPermanent ? Data_Dynamic : Data_Static );
            IOMan_BlockAddress         block;
            tsp00_PageNo               pageNo;
            
            if( bIsRestore )
                memcpy( &pageNo, pPage, sizeof( tsp00_PageNo ));
            else
                pageNo = messBlock.mb_qual()->mut_pno();
                        
            if( ! Converter_IDiagnose::Instance().GetBlockAddress(
                taskId, pageNo, addrMode, block ))
            {
                trError = e_no_converter_entry;
                return false;
            }
            if( bIsRestore )
            {
                if( ! IOMan_IDiagnose::GetInstance().WritePageDirectToDataVolume( 
                    taskId, block, pPage )){
                    return false;
                }
                return true;
            }
            // extract data page
            if( ! IOMan_IDiagnose::GetInstance().ReadPageDirectFromDataVolume( 
                taskId, block, pPage )){
                return false;
            }
            memcpy( messBlock.mb_data()->mbp_buf(), pPage, pageSize );
            messBlock.mb_data_len() = pageSize;
            bPageWasExtracted       = true;
            return true;
        }
    case diagFDir1_egg00:
        {
            const tsp00_PageNo fdir1Root = bd17GetFdirRoot();
            bd01ExtractTable( trans, fdir1Root, messBlock.mb_qual()->mut_hostfn());
        }
        break;
    case diagFDir2_egg00:
        {
            const tsp00_PageNo fdir2Root = bd17GetLongFdirRoot();
            bd01ExtractTable( trans, fdir2Root, messBlock.mb_qual()->mut_hostfn());
        }
        break;
    case diagFile_egg00:
        {
            tsp00_PageNo tableRoot = messBlock.mb_qual()->mut_pno();
            bd01ExtractTable( trans, tableRoot, messBlock.mb_qual()->mut_hostfn());
        }
        break;
    case diagInvRoot_egg00:
        {
            const tsp00_PageNo indexRoot = messBlock.mb_qual()->mut_pno();
            bd03ExtractIndex( trans, indexRoot, messBlock.mb_qual()->mut_hostfn());
        }
        break;
    case diagColRoot_egg00:
        {
            tsp00_PageNo blobRoot = messBlock.mb_qual()->mut_pno();
            bd05ExtractBlob( trans, blobRoot, messBlock.mb_qual()->mut_hostfn());
        }
        break;
    default:
        trError = e_not_implemented;
    }
    return( e_ok == trError );
}

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

02436 void Kernel_Administration::CreateSnapShot( tgg00_TransContext &trans )
{
    SAPDBTRACE_METHOD_DEBUG( "Kernel_Administration::CreateSnapShot", Common_Trace, 5 );
    
    Kernel_StateScope state( Kernel_State::criticalStateConfiguration, m_State );
    
    tsp00_TaskId        &taskId  = trans.trTaskId_gg00;
    tgg00_BasisError    &trError = trans.trError_gg00;

    if ( ! state.EnterCriticalState() )
      {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        trError = e_dbm_command_not_possible;
        return;
    }
    
    if( ! DropSnapShot( trans.trTaskId_gg00 )){
        trError = e_disk_not_accessible;
        return;
    }

    // The filesysten is started, because an unsused position for the
    // snapShot restart page is needed and therefore the converter
    // has to be started.

    if( ! RestartFilesystem(trans, false) )
      {
        Offline( trError );
    }
    
    SAPDBERR_ASSERT_STATE( NULL != k57restartrec );
        
    IOMan_BlockAddress  restartPageAddr;
    Kernel_RestartPage  restartPage( k57restartrec );

    k57restartrec->rstConfigParam_kb00().crSnapShotRestartRecordAddr_kb00  = 0;

    IOMan_IDataIO::GetInstance().WriteSnapShotRestartPage( taskId, restartPage, restartPageAddr );
      
    const SAPDB_Int                 volNoBitCount   = Converter_ICommon::Instance().GetVolumeNoBitCount();
    const IOMan_PackedBlockAddress  packedBlockAddr = restartPageAddr.Pack( volNoBitCount );

    k57restartrec->rstConfigParam_kb00().crSnapShotRestartRecordAddr_kb00 = packedBlockAddr;
        
    IOMan_IDataIO::GetInstance().WriteRestartPage( taskId, restartPage );
   
    b01shutdown_filesystem( trans );
}

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

02487 void Kernel_Administration::ActivateSnapShot( tgg00_TransContext &trans )
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::ActivateSnapShot", Common_Trace, 5);

    tsp00_TaskId        &taskId  = trans.trTaskId_gg00;
    tgg00_BasisError    &trError = trans.trError_gg00;
   
    Kernel_StateScope state1( Kernel_State::criticalStateRedo, m_State );
    Kernel_StateScope state2( Kernel_State::criticalStateBackup, m_State );
    Kernel_StateScope state3( Kernel_State::criticalStateConfiguration, m_State );
    
    if( ! state1.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Redo"));
        trError = e_dbm_command_not_possible;
        return;
    }
    if( ! state2.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Backup"));
        trError = e_dbm_command_not_possible;
        return;
    }
    if( ! state3.EnterCriticalState() )
    {
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        trError = e_dbm_command_not_possible;
        return;
    }
    if( ! ReadRestartRecord(taskId) )
    {
        trError = e_disk_not_accessible;
        return;
    }
    
    if( 0 == k57restartrec->rstConfigParam_kb00().crSnapShotRestartRecordAddr_kb00 ){
        trError = e_no_snapshot_available;
        return;
    }
    
    {
        const IOMan_PackedBlockAddress  packedBlockAddr( 
            k57restartrec->rstConfigParam_kb00().crSnapShotRestartRecordAddr_kb00 );
        const SAPDB_Int           volNoBitCount   = Converter_ICommon::Instance().GetVolumeNoBitCount();
        const IOMan_BlockAddress  restartPageAddr = packedBlockAddr.Unpack( volNoBitCount );
        
        IOMan_IDataManager &ioManData = IOMan_IDataManager::GetInstance();
        
        if ( ! ioManData.OpenOneDataVolume( taskId, restartPageAddr.GetDeviceNo()) ){
            trError = e_disk_not_accessible;
            return;
        }
        
        // read frozen info page into the memory of the old restart info page
        Kernel_RestartPage  snapShotRestartPage( k57restartrec );
        
        IOMan_IDataIO::GetInstance().ReadSnapShotRestartPage( taskId, 
            snapShotRestartPage, restartPageAddr );

        // write snapshot block address into the restart page, to
        // guarantee that the snapshot is still available.

        k57restartrec->rstConfigParam_kb00().crSnapShotRestartRecordAddr_kb00 = packedBlockAddr;

        ioManData.CloseAllDataVolumes( taskId );
    }
    
    if( ! FlushRestartRecord( taskId )){
        trError = e_disk_not_accessible;
        return;
    }

    ClearLogArea( taskId );
}

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

02564 void Kernel_Administration::DropSnapShot( tgg00_TransContext &trans )
{
    SAPDBTRACE_METHOD_DEBUG( "Kernel_Administration::DropSnapShot", Common_Trace, 5 );
    
    Kernel_StateScope state( Kernel_State::criticalStateConfiguration, m_State );
    
    if( ! state.EnterCriticalState()){
        RTE_Message(Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration"));
        trans.trError_gg00 = e_dbm_command_not_possible;
        return;
    }

    if( ! DropSnapShot( trans.trTaskId_gg00 )){
        trans.trError_gg00 = e_disk_not_accessible;
        return;
    }
}

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

02584 bool Kernel_Administration::DropSnapShot( const tsp00_TaskId    taskId )
{
    if( ! ReadRestartRecord( taskId ))
        return( false );

    if( 0 == k57restartrec->rstConfigParam_kb00().crSnapShotRestartRecordAddr_kb00 )
        return( true ); // no snapshot available

    k57restartrec->rstConfigParam_kb00().crSnapShotRestartRecordAddr_kb00 = 0;
        
    if( FlushRestartRecord( taskId ))
        return( true );

    return( false );
}

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

02602 bool Kernel_Administration::IsHotStandbyConfigured()
{
    SAPDBErr_MessageList                   errlist;
    RTEHSS_KernelInterface::Configuration  config;

    (void) RTEHSS_KernelInterface::Instance().
              CheckForValidHotStandbyConfiguration(config, errlist);

    if ( RTEHSS_KernelInterface::ConfigurationHotStandby == config )
        return true;
      else if ( RTEHSS_KernelInterface::ConfigurationNormal == config )
        return false;
      else // ConfigurationInvalid, ...
    {
        errlist = errlist + Admin_Exception(__CONTEXT__, KERNEL_HINT_HOTSTANDBY_CONFIG);
        RTE_Message(errlist);
        Offline();
    }
    return false;
}
/* --------------------------------------------------------------------------- */
02623 void Kernel_Administration::ForceHotStandbyRoleToMaster()
{
    SAPDBErr_MessageList errlist;

    if ( ! RTEHSS_KernelInterface::Instance().SetRoleToMaster(errlist) )
    {
        RTE_Message(errlist);
        Offline(true);
    }
    m_State.SetMaster();
}
/* --------------------------------------------------------------------------- */
02635 void Kernel_Administration::DetermineHotStandbyRole()
{
    if ( m_State.IsHotStandbyRoleSet()
         ||
         ! IsHotStandbyConfigured() )
        return;
        
    // hot stand by is configured - which role does the running instance own

    RTEHSS_KernelInterface &hss = RTEHSS_KernelInterface::Instance();
    SAPDBErr_MessageList    errlist;
      SAPDB_Char const *      localNode = RTE_ISystem::Instance().GetLocalNodeName();
    RTE_Nodename            masterName;
    bool                    shouldBeMaster;
    
    // Try to read the master node name from log info page

    if ( GetMaster (masterName, errlist) )
        shouldBeMaster = ! strcmp (masterName, localNode);
    else
    {
        SAPDBErr_MessageList    tmperrlist;
        strcpy (masterName, "NOT_INITIALIZED");
        if ( ! hss.IsDefaultMaster (shouldBeMaster, tmperrlist) )
        {
            RTE_Message(errlist+tmperrlist);
            Offline(true);
        }
    }
    
    if ( shouldBeMaster )
        ForceHotStandbyRoleToMaster();
    else
    {
        if ( ! hss.SetRoleToStandby(masterName, errlist) )
        {
            RTE_Message(errlist);
            Offline(true);
        }
        m_State.SetStandby();
    }
}
/* --------------------------------------------------------------------------- */
02678 bool Kernel_Administration::IsStandby()
{
    return m_State.IsStandby();
}
/* --------------------------------------------------------------------------- */
02683 bool Kernel_Administration::GetMaster (RTE_Nodename         &masterNodeName,
                                       SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::GetMaster", Common_Trace, 5);

    masterNodeName[0] = 0; // set to empty string

    tsp00_TaskId taskid;
        
    vgetpid(taskid);

    if ( m_State.IsAdmin() )
    {
        bool leaveCriticalState = false;

        if ( ! m_State.IsCriticalState(Kernel_State::criticalStateConfiguration) )
        {
            if ( ! m_State.EnterCriticalState(Kernel_State::criticalStateConfiguration) )
            {
                errlist = Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration");
                return false;
            }
            leaveCriticalState = true;
        }

        if ( ! Log_Volume::Instance().PrepareLogInfoPageForRead (taskid) )
        {
            if ( leaveCriticalState )
                m_State.LeaveCriticalState(Kernel_State::criticalStateConfiguration);
                
            errlist = Admin_Exception(__CONTEXT__,KERNEL_PREPARE_LOGINFO_PAGE_FOR_READ_FAILED);
            return false;
        }

        if ( leaveCriticalState )
            m_State.LeaveCriticalState(Kernel_State::criticalStateConfiguration);
    }

    return Log_Volume::Instance().GetMasterNodeName(masterNodeName);
}
/* --------------------------------------------------------------------------- */
Kernel_IAdminHotStandby::InitStandbyResult
02725 Kernel_Administration::InitStandby (tsp00_TaskId          taskid,
                                    SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::InitStandby", Common_Trace, 5);

    if ( ! m_State.IsStandby() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Init Standby","role is not standby");
        return InitStandbyWrongRole;
    }

    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);

    if ( ! state.EnterCriticalState() )
    {
        errlist = Admin_Exception(__CONTEXT__, KERNEL_ENTER_CRITICAL_STATE_FAILED, "Configuration");
        return InitStandbyNotAllowed;
    }

    RTEHSS_KernelInterface &hss = RTEHSS_KernelInterface::Instance();

    // Notify INIT STANDBY was accepted
    hss.GotInitStandby();

    // Check if the master wants to become a standby

    RTE_Nodename masterNodeName;
    RTE_Nodename ownNodeName;

    if ( ! GetMaster (masterNodeName, errlist) )
        return InitStandbyCannotReadMaster;

    if ( ! hss.GetOwnNodeName (ownNodeName,errlist) )
        return InitStandbyCannotReadOwnName;

    if ( ! strcmp (masterNodeName, ownNodeName) )
    {
        errlist = Admin_Exception(__CONTEXT__, KERNEL_INIT_STANDBY_NOT_ALLOWED,
                                  ownNodeName, masterNodeName);
        return InitStandbyNotAllowed;
    }
    
    // Write a message to knldiag
    
    RTE_Message(Admin_Exception(__CONTEXT__, KERNEL_INIT_STANDBY_START, masterNodeName));
    
    if( ! IOMan_IDataManager::GetInstance().CreateAllDataVolumes(taskid) )
        return InitStandbyCreateDataVolumesFailed;

    // determine the masters mapping info

    RTEHSS_KernelInterface::Mapping otherMapping;

    if ( ! hss.SendInfoStandby(masterNodeName, otherMapping, errlist) )
        return InitStandbyCommunicationError;

    // Establish the data volume mirror

    if ( ! hss.EstablishMirror(otherMapping, errlist) )
        return InitStandbyEstablishMirrorFailed;

      if ( ! hss.FreeMemory(otherMapping.mappingString, errlist) )
        return InitStandbyFreeMemoryFailed;

    // The data volumes become prepared for usage

    if ( ! hss.SendPrepareStandby(errlist) )
        return InitStandbyCommunicationError;
    
      IOMan_IDataManager::GetInstance().CloseAllDataVolumes(taskid);
    return InitStandbyOk;
}
/* --------------------------------------------------------------------------- */
Kernel_IAdminHotStandby::RestartStandbyResult
02799 Kernel_Administration::RestartStandby (tsp00_TaskId          taskid,
                                       tgg00_TransContext   &serverStartContext,
                                       SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RestartStandby", Common_Trace, 5);

    RTEHSS_KernelInterface &hss                = RTEHSS_KernelInterface::Instance();
    tgg00_BasisError       &trError            = serverStartContext.trError_gg00;
    
    if ( ! m_State.IsStandby() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Restart Standby","role is not standby");
        return RestartStandbyWrongRole;
    }

    if ( m_State.IsOnline() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Restart Standby","state is online");
        return RestartStandbyNotAllowed;
    }

    if ( m_State.IsRedo() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Restart Standby","state is redo");
        return RestartStandbyNotAllowed;
    }

    if ( ! m_State.EnterCriticalState(Kernel_State::criticalStateRedo) )
    {
        errlist = Admin_Exception(__CONTEXT__, KERNEL_ENTER_CRITICAL_STATE_FAILED, "Redo");
        return RestartStandbyNotAllowed;
    }

    // Notify RESTART STANDBY was accepted
    hss.GotRestartStandby();
    
    if ( ! RestartPrepare (serverStartContext, restartKindStandby, errlist) )
    {
        m_State.LeaveCriticalState(Kernel_State::criticalStateRedo);
        return RestartStandbyRestartPrepareFailed;
    }
    
    // START THE (REDO LOG) STANDBY SERVER
    
    tgg00_TransChild childtrans;
    tgg00_MessBlock  sendmblock;

    g01mblock_init (serverStartContext, m_restart, mm_standby, sendmblock);
    k53child_trans_build (m_restart, 0, childtrans);
    sendmblock.mb_reply() = false;
    
    k90send (sendmblock, childtrans);
    
    if ( sendmblock.mb_trns()->trError_gg00 != e_ok )
    {
        m_State.LeaveCriticalState(Kernel_State::criticalStateRedo);
        return RestartStandbyTriggerServerFailed;
    }

    return RestartStandbyOk;
}
/* --------------------------------------------------------------------------- */
Kernel_IAdminHotStandby::PrepareStandbyResult
02862 Kernel_Administration::PrepareStandby (      tsp00_TaskId          taskid,
                                             tgg00_TransContext   &savepointStartContext,
                                             SAPDBErr_MessageList &errlist,
                                       const SAPDB_Char           *newStandbyNode)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::PrepareStandby", Common_Trace, 5);

    if ( ! m_State.IsMaster() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Prepare Standby","role is not master");
        return PrepareStandbyWrongRole;
    }

    // communicate with standby to split the mirror of the data volumes
    RTEHSS_KernelInterface          &hss = RTEHSS_KernelInterface::Instance();
    RTEHSS_KernelInterface::Mapping  otherMapping;
    RTE_Nodename                     othernode;
    
    strcpy ( othernode, newStandbyNode );
    
    SAPDB_Int4 nodeIndex;

    if ( ! RTEHSS_StandbyNodeList::Instance().AddNode(othernode, nodeIndex) )
    {
        errlist = Admin_Exception(__CONTEXT__, KERNEL_STANDBY_NODELIST_OP_FAILED,
                                  "PREPARE STANDBY", othernode);
        return PrepareStandbyAddNodeFailed;
    }

    RTEHSS_StandbyNodeList::Instance().GotPrepareStandby(nodeIndex);

    // get mapping of standby node
    if ( ! hss.SendInfoStandby(othernode, otherMapping, errlist) )
    {
        errlist.AppendNewMessage( Admin_Exception(__CONTEXT__, KERNEL_STANDBY_SEND_TO_NODE_FAILED,
                                                  "PREPARE STANDBY", othernode) );
        return PrepareStandbySendInfoToStandbyFailed;
    }
    
    // Trigger savepoint to increase the chance for the standby to be able to restart
    // without the error: log and data incompatible
    Log_SavepointManager.StartSavepointAndWait (savepointStartContext, Log_SVPReasonSplitMirror);
    if ( savepointStartContext.trError_gg00 != e_ok )
        return PrepareStandbyStartSavepointFailed;
        
    // finally split the mirror to the standby which can run now autonomous
    if ( ! hss.SplitMirror(otherMapping, errlist) )
        return PrepareStandbySplitMirrorFailed;

    if ( ! hss.FreeMemory(otherMapping.mappingString, errlist) )
        return PrepareStandbyFreeMemoryFailed;
      
    return PrepareStandbyOk;
}
/* --------------------------------------------------------------------------- */
Kernel_IAdminHotStandby::TakeOverResult
02918 Kernel_Administration::TakeOver (tsp00_TaskId          taskid,
                                 Kernel_Date           date,
                                 Kernel_Time           time,
                                 SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::TakeOver", Common_Trace, 5);

    if ( ! m_State.IsStandby() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"TakeOver","role is not standby");
        return TakeOverWrongRole;
    }
    if ( m_State.IsOnline() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"TakeOver","state is already online");
        return TakeOverNotAllowed;
    }
    if ( ! m_State.IsRedo() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"TakeOver","state is not redo");
        return TakeOverNotAllowed;
    }
    if ( m_WaitingForTakeOverReady != cgg_nil_pid )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"TakeOver","it seems that another TakOver is running");
        return TakeOverNotAllowed;
    }
    if ( IsMasterAlive(errlist) ) // PTS 1127374 UH 2004-01-23
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"TakeOver","master is alive");
        return TakeOverNotAllowed;
    }

    // Notify TAKEOVER was accepted
    RTEHSS_KernelInterface::Instance().GotTakeover();

    // First make sure that no one is accessing the log volumes.
    m_RedoManager.SuspendRedoReader(taskid);
    
    // detach and attach again the log volume
    Log_Volume::Instance().CloseLogArea(taskid);

    if ( ! RTEHSS_KernelInterface::Instance().SetRoleToMaster(errlist) )
        return TakeOverSetRoleToMasterFailed;
        
    m_State.SetMaster();

    if ( ! Log_Volume::Instance().OpenLogArea(taskid) )
        return TakeOverOpenLogAreaFailed;
    
    // re-read log info page
    Log_Volume::Instance().ReReadLogInfoPageFromLogVolume(taskid);

    // Resume the redo log reader which reads now until the end of the log
    if ( m_State.IsRedo() )
    {
        m_WaitingForTakeOverReady = taskid;
        m_RedoManager.ResumeRedoReader(date,time);
        vsuspend(m_WaitingForTakeOverReady, KERNEL_ADMIN_WAIT_FOR_TAKEOVER);
        m_WaitingForTakeOverReady = cgg_nil_pid;
    }
    return TakeOverOk;
}
/* --------------------------------------------------------------------------- */
Kernel_IAdminHotStandby::RegisterStandbyResult
02983 Kernel_Administration::RegisterStandby (      tsp00_TaskId            taskid,
                                              SAPDBMem_IRawAllocator &allocator,
                                        const SAPDB_Char             *standbyNodeId,
                                              Log_RawDeviceOffset     firstNeededOffset,
                                              Log_RawDeviceOffset    &maxValidOffset,
                                              SAPDBErr_MessageList   &errlist )
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RegisterStandby", Common_Trace, 5);

    if ( ! m_State.IsMaster() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"RegisterStandby","role is not master");
        return RegisterStandbyWrongRole;
    }

    if ( ! m_State.IsOnline() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"RegisterStandby","state is not online");
        return RegisterStandbyNotAllowed;
    }

    SAPDB_Int4   nodeIndex;
    RTE_Nodename othernode;
    
    strcpy ( othernode, standbyNodeId );

    if ( ! RTEHSS_StandbyNodeList::Instance().FindNode(othernode, nodeIndex) )
    {
        // The standby did not begin with Init Standby (new mirror)
        // But instead the standby tried immediately the Restart Standby
        if ( ! RTEHSS_StandbyNodeList::Instance().AddNode(othernode, nodeIndex) )
            return RegisterStandbyAddNodeFailed;
    }
      
    RTEHSS_StandbyNodeList::Instance().GotRegisterStandby (nodeIndex, firstNeededOffset);
    
    // Tell the other standby's that there is a new one
    SAPDB_UInt propagateErrors = PropagateInfoAboutStandby (allocator, standbyNodeId, true, errlist);
    if ( propagateErrors > 0 )
        return RegisterStandbyPropagationFailed;
        
    if ( ! errlist.IsEmpty() )
    {
        RTE_Message(errlist);
        errlist.ClearMessageList();
    }

    RTE_Message(Admin_Exception(__CONTEXT__, KERNEL_NEW_STANDBY,
                                othernode, SAPDB_ToString(firstNeededOffset)));

    // Give the new standby the last readable offset for his Redo. This is not the current writePosition,
    // but the page before this, because the last page may be written multiple times.
    // m_LastSyncOffset = (--Log_Volume::Instance().GetWriterPosition()).GetPosition();
    m_LastSyncOffset = Log_Savepoint::GetLastSavepointOffset(); // UH 2003-11-27
    maxValidOffset   = m_LastSyncOffset;
    
    // Add the new standby to the standby list
    RTE_Nodename newStandbyNode;
    strcpy ( newStandbyNode, standbyNodeId );

    RTEHSS_StandbyNodeList::Instance().SetWaitForSynchronize(nodeIndex);

    m_SynchronizationIsNeeded = true;

    return RegisterStandbyOk;
}
/* --------------------------------------------------------------------------- */
Kernel_IAdminHotStandby::InsertStandbyResult
03051 Kernel_Administration::InsertStandby (      tsp00_TaskId          taskid,
                                      const SAPDB_Char           *standbyNodeId,
                                            SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::InsertStandby", Common_Trace, 5);

    if ( ! m_State.IsStandby() && ! m_State.IsMaster() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"InsertStandby","role is neither master nor standby");
        return InsertStandbyWrongRole;
    }

    // PTS 1125942 mb 2004-02-13 synchronize insert/remove-standby by using state-information
    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);
    if ( ! state.EnterCriticalState() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration");
        return InsertStandbyNotAllowed;
    }

    RTE_Nodename newStandbyNode;
    
    strcpy ( newStandbyNode, standbyNodeId );

    SAPDB_Int4 nodeIndex;
    
    if ( ! RTEHSS_StandbyNodeList::Instance().AddNode(newStandbyNode, nodeIndex) )
    {
        errlist = Admin_Exception(__CONTEXT__, KERNEL_STANDBY_NODELIST_OP_FAILED,
                                  "INSERT STANDBY", newStandbyNode);
        return InsertStandbyAddNodeFailed;
    }
        
    RTEHSS_StandbyNodeList::Instance().SetWaitForSynchronize(nodeIndex);
    
    return InsertStandbyOk;
}
/* --------------------------------------------------------------------------- */
Kernel_IAdminHotStandby::RemoveStandbyResult
03090 Kernel_Administration::RemoveStandby (      tsp00_TaskId          taskid,
                                      const SAPDB_Char           *standbyNodeId,
                                            SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RemoveStandby", Common_Trace, 5);

    if ( ! m_State.IsStandby() && ! m_State.IsMaster() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"RemoveStandby","role is neither master nor standby");
        return RemoveStandbyWrongRole;
    }

    // PTS 1125942 mb 2004-02-13 synchronize insert/remove-standby by using state-information
    Kernel_StateScope state (Kernel_State::criticalStateConfiguration, m_State);
    if ( ! state.EnterCriticalState() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_CRITICAL_STATE_NOT_ENTERED,"Configuration");
        return RemoveStandbyNotAllowed;
    }
    

    RTE_Nodename remStandbyNode;
    
    strcpy ( remStandbyNode, standbyNodeId );

    SAPDB_Int4 nodeIndex;

    if ( ! RTEHSS_StandbyNodeList::Instance().FindNode(remStandbyNode, nodeIndex) )
    {
        errlist = Admin_Exception(__CONTEXT__, KERNEL_STANDBY_NODELIST_OP_FAILED,
                                  "REMOVE STANDBY", remStandbyNode);
        return RemoveStandbyNodeNotFound;
    }
      
    RTEHSS_StandbyNodeList::Instance().RemoveNode(nodeIndex);

    // tell all the other standby nodes that current standby node is not available
    SAPDB_UInt propagateErrors = PropagateInfoAboutStandby(m_Allocator, remStandbyNode, false, errlist);

    // if neccessary switch off synchronization
    if (RTEHSS_StandbyNodeList::Instance().IsEmpty())
    {
        m_SynchronizationIsNeeded  = false;
        m_LastSyncOffset.Invalidate();
        m_LastSyncTime.Invalidate();
        RTE_Message(Admin_Exception(__CONTEXT__, KERNEL_LAST_STANDBY_REMOVED,standbyNodeId));
    }
    else
    {
        RTE_Message(Admin_Exception(__CONTEXT__, KERNEL_STANDBY_REMOVED,standbyNodeId));
    }

    return RemoveStandbyOk;
}
/* --------------------------------------------------------------------------- */
Kernel_IAdminHotStandby::SynchronizeResult
03146 Kernel_Administration::Synchronize (tsp00_TaskId          taskid,
                                    Log_RawDeviceOffset   newMaxOffset,
                                    Log_RawDeviceOffset  &lastRedoOffset,
                                    SAPDBErr_MessageList &errlist)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::Synchronize", Common_Trace, 5);

    lastRedoOffset.Invalidate();
    
    if ( ! m_State.IsStandby() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Synchronize","role is not standby");
        return SynchronizeWrongRole;
    }
    if ( ! m_State.IsRedo() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Synchronize","state is not redo");
        return SynchronizeNotAllowed;
    }
    if ( m_State.IsOnline() )
    {
        errlist = Admin_Exception(__CONTEXT__,KERNEL_COMMAND_NOT_ALLOWED,"Synchronize","state is online");
        return SynchronizeNotAllowed;
    }

    RTE_Message(Admin_Exception(__CONTEXT__, KERNEL_SYNC_RECEIVED,SAPDB_ToString(newMaxOffset)));

    m_RedoManager.SetLastRedoReadOffset(newMaxOffset, taskid);
    
    lastRedoOffset = m_RedoManager.GetLastRedoReadOffset();

    RTEHSS_KernelInterface::Instance().GotSynchronize(newMaxOffset, lastRedoOffset);
    return SynchronizeOk;
}
/* --------------------------------------------------------------------------- */
Kernel_IAdminHotStandby::TriggerSynchronizeResult
03182 Kernel_Administration::TriggerSynchronize (tgg00_TransContext &serverTaskStartContext)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::TriggerSynchronize", Common_Trace, 5);

    if ( ! m_SynchronizationIsNeeded || m_SynchronizationIsRunning )
        return TriggerSynchronizeOk; // fast return because called by timeout task
    
    // needs no critical region because is only called by timeout task
    m_SynchronizationIsRunning = true;
    
    Log_RawDeviceIterator currentWriterPosition = Log_Volume::Instance().GetWriterPosition();

    if ( m_LastSyncOffset.IsValid()
         &&
         currentWriterPosition.GetPosition() == m_LastSyncOffset )
      {
        // Not enough log was written to sync again.
        m_SynchronizationIsRunning = false;
        return TriggerSynchronizeNotNeeded;
      }

    // decrement the position, because the page on the current write position must not been read by the standby
    --currentWriterPosition;

    if ( m_LastSyncOffset.IsValid()
         &&
         currentWriterPosition.GetDistanceFrom(m_LastSyncOffset) < 1 )
      {
        // Not enough log was written to sync again.
        m_SynchronizationIsRunning = false;
        return TriggerSynchronizeNotNeeded;
      }
    
    if (m_LastSyncTime.IsValid())           // PTS 1125481 mb 2003-11-18
    {
        RTEConf_Parameter::Integer syncInterval = 0;
        SAPDBErr_MessageList       error;
        if( ! RTEConf_ParameterAccess::Instance()->GetInteger(UTF8("HS_SYNC_INTERVAL"), syncInterval, error) )
        {
            RTE_Message (error);
            RTE_Crash(SAPDBErr_Exception(__CONTEXT__,SAPDBERR_ASSERT_STATE_FAILED,"GetParameter: HS_SYNC_INTERVAL"));
        }

        Kernel_DateTime startOfSyncInterval;
        startOfSyncInterval.DetermineDelayedTime(static_cast<SAPDB_UInt>(syncInterval));
        
        if (startOfSyncInterval.OlderThan(m_LastSyncTime))
        {
            // span of time since last synchronize not long enough
            m_SynchronizationIsRunning = false;
            return TriggerSynchronizeNotNeeded;
        }
    }
        
    // Server is started now

    tgg00_TransChild childtrans;
    tgg00_MessBlock  sendmblock;

    g01mblock_init (serverTaskStartContext, m_restart, mm_consistent, sendmblock);
    k53child_trans_build (m_restart, 0, childtrans);
    sendmblock.mb_reply() = false;
    
    k90send (sendmblock, childtrans);
    
    if ( sendmblock.mb_trns()->trError_gg00 != e_ok )
      {
        m_SynchronizationIsRunning = false;
        return TriggerSynchronizeStartSyncServerFailed;
      }
    
    return TriggerSynchronizeOk;
}
/* --------------------------------------------------------------------------- */
03256 void Kernel_Administration::SynchronizeServer ( tsp00_TaskId            taskid,
                                                SAPDBMem_IRawAllocator &allocator)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::SynchronizeServer", Common_Trace, 5);

    RTEHSS_KernelInterface &hss = RTEHSS_KernelInterface::Instance();
    SAPDBErr_MessageList    errlist;
    RTEHSS_NodeChain        nodechain (allocator);
    RTE_Nodename            sendToNodename;
    Log_RawDeviceIterator   lastValidReadPosition = Log_Volume::Instance().GetWriterPosition();
    Log_RawDeviceOffset     firstStillNeededOffset;
    Log_RawDeviceOffset     offsetFromStandby;

    if ( m_LastSyncOffset == lastValidReadPosition.GetPosition() )
    {
        RTE_Message(SAPDBErr_Exception(__CONTEXT__,SAPDBERR_ASSERT_STATE_FAILED,
                    "Kernel_Administration::SynchronizeServer: m_LastSyncOffset == lastValidReadPosition") );
        return;
    }

    // decrement the position, because the page on the current write position must not been read by the standby
    --lastValidReadPosition;
    m_LastSyncOffset = lastValidReadPosition.GetPosition();
    m_LastSyncTime.DetermineTime(); 
    
    if ( ! nodechain.Fill() )
    {
        RTE_Message(SAPDBErr_Exception(__CONTEXT__,SAPDBERR_ASSERT_STATE_FAILED,
                    "Kernel_Administration::SynchronizeServer: nodechain.Fill() failed => NO SYNC WITH STANDBY") );
        return;
    }
    
    if ( ! nodechain.FirstNode(sendToNodename) )
    {
        // no standby nodes available - this should not happen
        RTE_Message(SAPDBErr_Exception(__CONTEXT__,SAPDBERR_ASSERT_STATE_FAILED,
                    "Kernel_Administration::SynchronizeServer: no standby node found. Synchronization switched off!") );
        m_SynchronizationIsNeeded  = false;
        m_SynchronizationIsRunning = false;
        m_LastSyncTime.Invalidate();
        m_LastSyncOffset.Invalidate();
        return;
    }
    
    RTEHSS_KernelInterface::LogPosition auxPosition;

    do
    {
        if ( ! hss.SendSynchronize(sendToNodename, lastValidReadPosition.GetPosition(), auxPosition, errlist) )
        {
            if ( Common_Trace.TracesLevel(6) )
                Kernel_VTrace() << "Sync: '"   << sendToNodename
                                << "' send: " << lastValidReadPosition.GetPosition()
                                << ", received: NO ANSWER";

            if ( ! errlist.IsEmpty() )
            {
                RTE_Message(errlist);
                errlist.ClearMessageList();
            }

            // remove the not answering standby node from nodelist.
            RemoveStandby (taskid, sendToNodename, errlist); // PTS 1125942 mb 2004-01-14
            if ( ! errlist.IsEmpty() )
            {
                RTE_Message(errlist);
                errlist.ClearMessageList();
            }
        }
        else
        {
            if ( Common_Trace.TracesLevel(6) )
                Kernel_VTrace() << "Sync: '"   << sendToNodename
                                << "' send: " << lastValidReadPosition.GetPosition()
                                << ", received: " << firstStillNeededOffset;
            offsetFromStandby = auxPosition;
            RTE_Message(Admin_Exception(__CONTEXT__, KERNEL_SYNCREPLY_RECEIVED,
                                        sendToNodename, SAPDB_ToString(offsetFromStandby)));

            if ( firstStillNeededOffset.IsInvalid() )
                firstStillNeededOffset = offsetFromStandby;
            else
            {
                if ( offsetFromStandby.IsValid()
                     &&
                     lastValidReadPosition.GetDistanceFrom(offsetFromStandby) 
                     >
                     lastValidReadPosition.GetDistanceFrom(firstStillNeededOffset) )
                {
                    firstStillNeededOffset = offsetFromStandby;
                }
            }
        }
    }
    while ( nodechain.NextNode(sendToNodename) );

    m_SynchronizationIsRunning = false;
}
/* --------------------------------------------------------------------------- */
03355 void Kernel_Administration::BroadcastLineToStandbys (
                            const BroadcastDestination    dest,
                                  tsp00_TaskId            taskid,
                            const char  *                 line,
                            const SAPDB_UInt              lineLength)
                        
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::BroadcastLineToStandbys", Common_Trace, 5);

    RTEHSS_KernelInterface &hss = RTEHSS_KernelInterface::Instance();
    SAPDBErr_MessageList    errlist;
    RTEHSS_NodeChain        nodechain (m_Allocator);
    RTE_Nodename            sendToNodename;

    if (!IsHotStandbyConfigured())
    {
        return;
    }
    if ( ! nodechain.Fill() )
    {
        RTE_Message(SAPDBErr_Exception(__CONTEXT__,SAPDBERR_ASSERT_STATE_FAILED,
                    "Kernel_Administration::BroadcastLineToStandbys: nodechain.Fill() failed => NO BackupHistory sent") );
        return;
    }
    
    if ( ! nodechain.FirstNode(sendToNodename) )
    {
        // no standby nodes available - this should not happen, but this is no error
        return;
    }
    
    // create the string, which has to be send
    char * lineToSend;
    newarray (lineToSend, lineLength + 3, m_Allocator);
    lineToSend[0] = '\'';
    memcpy(&lineToSend[1], line, lineLength);
    lineToSend[lineLength+1] =  '\'';
    lineToSend[lineLength+2] =  '\0';
    
    do
    {
        bool sendWasOk;
        if (dest == BackupHistoryFile)
        {
            sendWasOk = hss.SendBackupHistoryInfo(sendToNodename, 
                                                  lineToSend,
                                                  errlist);
        }
        else
        {
            sendWasOk = hss.SendBackupMediumInfo(sendToNodename, 
                                                 lineToSend,
                                                 errlist);
        }
        if ( ! sendWasOk )
        {
            if ( Common_Check.ChecksLevel(6) )
                Kernel_VTrace() << "BroadcastLineToStandbys: '"   << sendToNodename
                                << ", received: NO ANSWER";

            if ( ! errlist.IsEmpty() )
            {
                RTE_Message(errlist);
                errlist.ClearMessageList();
            }

            // remove the not answering standby node from nodelist.
            RemoveStandby (taskid, sendToNodename, errlist);
            if ( ! errlist.IsEmpty() )
            {
                RTE_Message(errlist);
                errlist.ClearMessageList();
            }
        }
        else
        {
            if ( Common_Check.ChecksLevel(6) )
                Kernel_VTrace() << "BroadcastLineToStandbys: '"   << sendToNodename;
        }
    }
    while ( nodechain.NextNode(sendToNodename) );
    
    destroyarray (lineToSend, lineLength + 3, m_Allocator);
}

/* --------------------------------------------------------------------------- */
03441 void Kernel_Administration::BroadcastBackupHistoryEntry (
                        tsp00_TaskId            taskid,
                        const char * line,
                        const SAPDB_UInt lineLength)
{
    BroadcastLineToStandbys (BackupHistoryFile, taskid, line, lineLength);
}
/* --------------------------------------------------------------------------- */
03449 void Kernel_Administration::BroadcastBackupMediumDefEntry (
                        tsp00_TaskId            taskid,
                        const char * line,
                        const SAPDB_UInt lineLength)
{
    BroadcastLineToStandbys (BackupMediumDefinitionFile, taskid, line, lineLength);
}
/* --------------------------------------------------------------------------- */
03457 void Kernel_Administration::RestartStandbyServer (tgg00_TransContext &restartContext)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::RestartStandbyServer", Common_Trace, 5);

    // go to permanent redo until Offline() or TakeOver() is called
    
    RestartIntern (restartContext, restartKindStandby, Kernel_Date(), Kernel_Time(), Log_IOSequenceNo());

    if ( restartContext.trError_gg00 != e_ok )
        Offline(restartContext.trError_gg00, true); // this should not be called

    if ( m_WaitingForTakeOverReady != cgg_nil_pid )
        vresume(m_WaitingForTakeOverReady);

    m_State.LeaveCriticalState(Kernel_State::criticalStateRedo);
}
/* --------------------------------------------------------------------------- */
SAPDB_UInt
03475 Kernel_Administration::PropagateInfoAboutStandby (       SAPDBMem_IRawAllocator &allocator,
                                                   const SAPDB_Char             *standbyNodename,
                                                         bool                    isActive,
                                                         SAPDBErr_MessageList   &errlist )
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::PropagateInfoAboutStandby", Common_Trace, 5);

    RTEHSS_NodeChain        nodechain (allocator);
    RTEHSS_KernelInterface &hss = RTEHSS_KernelInterface::Instance();
    RTE_Nodename            nodename;
    RTE_Nodename            sendToNodename;
    SAPDB_UInt              numErrorsOccured = 0;
    
    if ( strlen(standbyNodename)+1 > sizeof(nodename) )
        return 1;
        
    if ( ! nodechain.Fill() )
        return 1;
    
    if ( Common_Trace.TracesLevel(6) )
        Kernel_VTrace() << "Propagate that '" << standbyNodename
                        << "' is "            << (isActive?"ACTIVE":"INACTIVE") << FlushLine;
                    
    strcpy ( nodename, standbyNodename );

    if ( nodechain.FirstNode(sendToNodename) )
    {
        do
        {
            if ( strcmp(sendToNodename, nodename) )
            {
                if ( Common_Trace.TracesLevel(6) )
                    Kernel_VTrace() << "To: '"   << sendToNodename << "'" << FlushLine;
                if ( (isActive ? hss.SendInsertStandby (sendToNodename, nodename, errlist)
                               : hss.SendRemoveStandby (sendToNodename, nodename, errlist)) )
                {
                    ++numErrorsOccured;
                }
            }
        }
        while ( nodechain.NextNode(sendToNodename) );
    }
    return numErrorsOccured;
}

/* --------------------------------------------------------------------------- */
03521 void Kernel_Administration::GetDelayTimeStamp (Kernel_Date &date,
                                               Kernel_Time &time)
{
    SAPDBTRACE_METHOD_DEBUG ("Kernel_Administration::GetDelayTimeStamp", Common_Trace, 5);

    SAPDBErr_MessageList errlist;
    SAPDB_Int4           delayTimeInSec; // (see cserv.pcf)

    if ( ! RTEHSS_KernelInterface::Instance().GetOwnDelayTime (delayTimeInSec, errlist) )
        Offline(true);
    
    Kernel_DateTime dateTime;

    dateTime.DetermineDelayedTime( delayTimeInSec );
    date = dateTime.GetDate();
    time = dateTime.GetTime();
}

Generated by  Doxygen 1.6.0   Back to index